X-Git-Url: https://www.ginac.de/ginac.git//ginac.git?p=ginac.git;a=blobdiff_plain;f=ginac%2Fexpairseq.cpp;h=fe09681496b98f2513e1e70a55e5dfc421667ac4;hp=c18fadecc9934f5ad55de6d5bc59ca91549aba17;hb=df27e9de2a3f09aa315d1fe9fb90a3e5a5751989;hpb=e7cc6a764ff67b5885d6633385fac23ccc1dc9a7 diff --git a/ginac/expairseq.cpp b/ginac/expairseq.cpp index c18fadec..fe096814 100644 --- a/ginac/expairseq.cpp +++ b/ginac/expairseq.cpp @@ -3,7 +3,7 @@ * Implementation of sequences of expression pairs. */ /* - * GiNaC Copyright (C) 1999-2002 Johannes Gutenberg University Mainz, Germany + * GiNaC Copyright (C) 1999-2003 Johannes Gutenberg University Mainz, Germany * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by @@ -33,6 +33,7 @@ #include "wildcard.h" #include "print.h" #include "archive.h" +#include "operators.h" #include "utils.h" #if EXPAIRSEQ_USE_HASHTAB @@ -42,7 +43,7 @@ namespace GiNaC { -GINAC_IMPLEMENT_REGISTERED_CLASS_NO_CTORS(expairseq, basic) +GINAC_IMPLEMENT_REGISTERED_CLASS(expairseq, basic) ////////// // helper classes @@ -58,31 +59,23 @@ public: }; ////////// -// default ctor, dtor, copy ctor, assignment operator and helpers +// default constructor ////////// // public -expairseq::expairseq(const expairseq &other) -{ - copy(other); -} - -const expairseq &expairseq::operator=(const expairseq &other) -{ - if (this != &other) { - destroy(true); - copy(other); - } - return *this; -} +expairseq::expairseq() : inherited(TINFO_expairseq) +#if EXPAIRSEQ_USE_HASHTAB + , hashtabsize(0) +#endif // EXPAIRSEQ_USE_HASHTAB +{} // protected +#if 0 /** For use by copy ctor and assignment operator. */ void expairseq::copy(const expairseq &other) { - inherited::copy(other); seq = other.seq; overall_coeff = other.overall_coeff; #if EXPAIRSEQ_USE_HASHTAB @@ -104,11 +97,10 @@ void expairseq::copy(const expairseq &other) } #endif // EXPAIRSEQ_USE_HASHTAB } - -DEFAULT_DESTROY(expairseq) +#endif ////////// -// other ctors +// other constructors ////////// expairseq::expairseq(const ex &lh, const ex &rh) : inherited(TINFO_expairseq) @@ -126,6 +118,7 @@ expairseq::expairseq(const exvector &v) : inherited(TINFO_expairseq) expairseq::expairseq(const epvector &v, const ex &oc) : inherited(TINFO_expairseq), overall_coeff(oc) { + GINAC_ASSERT(is_a(oc)); construct_from_epvector(v); GINAC_ASSERT(is_canonical()); } @@ -134,6 +127,7 @@ expairseq::expairseq(epvector *vp, const ex &oc) : inherited(TINFO_expairseq), overall_coeff(oc) { GINAC_ASSERT(vp!=0); + GINAC_ASSERT(is_a(oc)); construct_from_epvector(*vp); delete vp; GINAC_ASSERT(is_canonical()); @@ -143,7 +137,7 @@ expairseq::expairseq(epvector *vp, const ex &oc) // archiving ////////// -expairseq::expairseq(const archive_node &n, const lst &sym_lst) : inherited(n, sym_lst) +expairseq::expairseq(const archive_node &n, lst &sym_lst) : inherited(n, sym_lst) #if EXPAIRSEQ_USE_HASHTAB , hashtabsize(0) #endif @@ -156,7 +150,11 @@ expairseq::expairseq(const archive_node &n, const lst &sym_lst) : inherited(n, s else break; } + n.find_ex("overall_coeff", overall_coeff, sym_lst); + + canonicalize(); + GINAC_ASSERT(is_canonical()); } void expairseq::archive(archive_node &n) const @@ -179,14 +177,9 @@ DEFAULT_UNARCHIVE(expairseq) // public -basic *expairseq::duplicate() const -{ - return new expairseq(*this); -} - void expairseq::print(const print_context &c, unsigned level) const { - if (is_of_type(c, print_tree)) { + if (is_a(c)) { unsigned delta_indent = static_cast(c).delta_indent; @@ -194,8 +187,8 @@ void expairseq::print(const print_context &c, unsigned level) const << std::hex << ", hash=0x" << hashvalue << ", flags=0x" << flags << std::dec << ", nops=" << nops() << std::endl; - unsigned num = seq.size(); - for (unsigned i=0; i(pattern.op(i))) { has_global_wildcard = true; global_wildcard = pattern.op(i); break; @@ -349,12 +337,12 @@ bool expairseq::match(const ex & pattern, lst & repl_lst) const // Chop into terms exvector ops; ops.reserve(nops()); - for (unsigned i=0; ireserve(num); - for (unsigned i=0; ipush_back(split_ex_to_pair(ops[i])); ex rest = thisexpairseq(vp, default_overall_coeff()); - for (unsigned i=0; iop(0).is_equal(global_wildcard)) + return rest.is_equal(it->op(1)); } repl_lst.append(global_wildcard == rest); return true; @@ -398,13 +386,15 @@ found: ; return inherited::match(pattern, repl_lst); } -ex expairseq::subs(const lst &ls, const lst &lr, bool no_pattern) const +ex expairseq::subs(const exmap & m, unsigned options) const { - epvector *vp = subschildren(ls, lr, no_pattern); + epvector *vp = subschildren(m, options); if (vp) - return ex_to(thisexpairseq(vp, overall_coeff)).basic::subs(ls, lr, no_pattern); + return ex_to(thisexpairseq(vp, overall_coeff)); + else if ((options & subs_options::algebraic) && is_exactly_a(*this)) + return static_cast(this)->algebraic_subs_mul(m, options); else - return basic::subs(ls, lr, no_pattern); + return subs_one_level(m, options); } // protected @@ -539,30 +529,28 @@ bool expairseq::is_equal_same_type(const basic &other) const #endif // EXPAIRSEQ_USE_HASHTAB } -unsigned expairseq::return_type(void) const +unsigned expairseq::return_type() const { return return_types::noncommutative_composite; } -unsigned expairseq::calchash(void) const +unsigned expairseq::calchash() const { unsigned v = golden_ratio_hash(this->tinfo()); - epvector::const_iterator i = seq.begin(), end = seq.end(); + epvector::const_iterator i = seq.begin(); + const epvector::const_iterator end = seq.end(); while (i != end) { -#if !EXPAIRSEQ_USE_HASHTAB - v = rotate_left_31(v); // rotation would spoil commutativity -#endif // EXPAIRSEQ_USE_HASHTAB v ^= i->rest.gethash(); #if !EXPAIRSEQ_USE_HASHTAB - v = rotate_left_31(v); + // rotation spoils commutativity! + v = rotate_left(v); v ^= i->coeff.gethash(); -#endif // EXPAIRSEQ_USE_HASHTAB +#endif // !EXPAIRSEQ_USE_HASHTAB ++i; } - + v ^= overall_coeff.gethash(); - v &= 0x7FFFFFFFU; - + // store calculated hash value only if object is already evaluated if (flags &status_flags::evaluated) { setflag(status_flags::hash_calculated); @@ -679,7 +667,7 @@ bool expairseq::expair_needs_further_processing(epp it) return false; } -ex expairseq::default_overall_coeff(void) const +ex expairseq::default_overall_coeff() const { return _ex0; } @@ -775,8 +763,8 @@ void expairseq::construct_from_2_ex(const ex &lh, const ex &rh) hashtabsize = 0; #endif // EXPAIRSEQ_USE_HASHTAB - if (is_ex_exactly_of_type(lh,numeric)) { - if (is_ex_exactly_of_type(rh,numeric)) { + if (is_exactly_a(lh)) { + if (is_exactly_a(rh)) { combine_overall_coeff(lh); combine_overall_coeff(rh); } else { @@ -784,7 +772,7 @@ void expairseq::construct_from_2_ex(const ex &lh, const ex &rh) seq.push_back(split_ex_to_pair(rh)); } } else { - if (is_ex_exactly_of_type(rh,numeric)) { + if (is_exactly_a(rh)) { combine_overall_coeff(rh); seq.push_back(split_ex_to_pair(lh)); } else { @@ -871,7 +859,7 @@ void expairseq::construct_from_expairseq_ex(const expairseq &s, const ex &e) { combine_overall_coeff(s.overall_coeff); - if (is_ex_exactly_of_type(e,numeric)) { + if (is_exactly_a(e)) { combine_overall_coeff(e); seq = s.seq; return; @@ -996,7 +984,7 @@ void expairseq::make_flat(const exvector &v) ++cit_s; } } else { - if (is_ex_exactly_of_type(*cit,numeric)) + if (is_exactly_a(*cit)) combine_overall_coeff(*cit); else seq.push_back(split_ex_to_pair(*cit)); @@ -1055,53 +1043,54 @@ void expairseq::make_flat(const epvector &v) } /** Brings this expairseq into a sorted (canonical) form. */ -void expairseq::canonicalize(void) +void expairseq::canonicalize() { - std::sort(seq.begin(), seq.end(), expair_is_less()); + std::sort(seq.begin(), seq.end(), expair_rest_is_less()); } /** Compact a presorted expairseq by combining all matching expairs to one * each. On an add object, this is responsible for 2*x+3*x+y -> 5*x+y, for * instance. */ -void expairseq::combine_same_terms_sorted_seq(void) +void expairseq::combine_same_terms_sorted_seq() { + if (seq.size()<2) + return; + bool needs_further_processing = false; - - if (seq.size()>1) { - epvector::iterator itin1 = seq.begin(); - epvector::iterator itin2 = itin1+1; - epvector::iterator itout = itin1; - epvector::iterator last = seq.end(); - // must_copy will be set to true the first time some combination is - // possible from then on the sequence has changed and must be compacted - bool must_copy = false; - while (itin2!=last) { - if (itin1->rest.compare(itin2->rest)==0) { - itin1->coeff = ex_to(itin1->coeff). - add_dyn(ex_to(itin2->coeff)); - if (expair_needs_further_processing(itin1)) - needs_further_processing = true; - must_copy = true; - } else { - if (!ex_to(itin1->coeff).is_zero()) { - if (must_copy) - *itout = *itin1; - ++itout; - } - itin1 = itin2; + + epvector::iterator itin1 = seq.begin(); + epvector::iterator itin2 = itin1+1; + epvector::iterator itout = itin1; + epvector::iterator last = seq.end(); + // must_copy will be set to true the first time some combination is + // possible from then on the sequence has changed and must be compacted + bool must_copy = false; + while (itin2!=last) { + if (itin1->rest.compare(itin2->rest)==0) { + itin1->coeff = ex_to(itin1->coeff). + add_dyn(ex_to(itin2->coeff)); + if (expair_needs_further_processing(itin1)) + needs_further_processing = true; + must_copy = true; + } else { + if (!ex_to(itin1->coeff).is_zero()) { + if (must_copy) + *itout = *itin1; + ++itout; } - ++itin2; - } - if (!ex_to(itin1->coeff).is_zero()) { - if (must_copy) - *itout = *itin1; - ++itout; + itin1 = itin2; } - if (itout!=last) - seq.erase(itout,last); + ++itin2; } - + if (!ex_to(itin1->coeff).is_zero()) { + if (must_copy) + *itout = *itin1; + ++itout; + } + if (itout!=last) + seq.erase(itout,last); + if (needs_further_processing) { epvector v = seq; seq.clear(); @@ -1120,7 +1109,7 @@ unsigned expairseq::calc_hashtabsize(unsigned sz) const size = nearest_power_of_2/hashtabfactor; if (size(e)) { hashindex = hashmask; } else { - hashindex = hash &hashmask; + hashindex = e.gethash() & hashmask; // last hashtab entry is reserved for numerics if (hashindex==hashmask) hashindex = 0; } - GINAC_ASSERT(hashindex>=0); GINAC_ASSERT((hashindexrest,numeric)) { + if (is_exactly_a(current->rest)) { --first_numeric; iter_swap(current,first_numeric); } else { @@ -1293,7 +1280,7 @@ void expairseq::drop_coeff_0_terms(epvector::iterator &first_numeric, // move terms with coeff 0 to end and remove them from hashtab // check only those elements which have been touched epp current = seq.begin(); - unsigned i = 0; + size_t i = 0; while (current!=first_numeric) { if (!touched[i]) { ++current; @@ -1340,7 +1327,7 @@ void expairseq::drop_coeff_0_terms(epvector::iterator &first_numeric, /** True if one of the coeffs vanishes, otherwise false. * This would be an invariant violation, so this should only be used for * debugging purposes. */ -bool expairseq::has_coeff_0(void) const +bool expairseq::has_coeff_0() const { epvector::const_iterator i = seq.begin(), end = seq.end(); while (i != end) { @@ -1363,7 +1350,7 @@ void expairseq::add_numerics_to_hashtab(epvector::iterator first_numeric, } } -void expairseq::combine_same_terms(void) +void expairseq::combine_same_terms() { // combine same terms, drop term with coeff 0, move numerics to end @@ -1389,7 +1376,7 @@ void expairseq::combine_same_terms(void) epvector::iterator first_numeric = seq.end(); epvector::iterator last_non_zero = seq.end()-1; - unsigned num = seq.size(); + size_t num = seq.size(); std::vector touched(num); unsigned number_of_zeroes = 0; @@ -1435,11 +1422,11 @@ bool expairseq::is_canonical() const epvector::const_iterator it_last = it; for (++it; it!=itend; it_last=it, ++it) { if (!(it_last->is_less(*it) || it_last->is_equal(*it))) { - if (!is_ex_exactly_of_type(it_last->rest,numeric) || - !is_ex_exactly_of_type(it->rest,numeric)) { + if (!is_exactly_a(it_last->rest) || + !is_exactly_a(it->rest)) { // double test makes it easier to set a breakpoint... - if (!is_ex_exactly_of_type(it_last->rest,numeric) || - !is_ex_exactly_of_type(it->rest,numeric)) { + if (!is_exactly_a(it_last->rest) || + !is_exactly_a(it->rest)) { printpair(std::clog, *it_last, 0); std::clog << ">"; printpair(std::clog, *it, 0); @@ -1559,28 +1546,32 @@ epvector * expairseq::evalchildren(int level) const * @see expairseq::subs() * @return pointer to epvector containing pairs after application of subs, * or NULL pointer if no members were changed. */ -epvector * expairseq::subschildren(const lst &ls, const lst &lr, bool no_pattern) const +epvector * expairseq::subschildren(const exmap & m, unsigned options) const { - GINAC_ASSERT(ls.nops()==lr.nops()); - - // The substitution is "complex" when any of the objects to be substituted - // is a product or power. In this case we have to recombine the pairs - // because the numeric coefficients may be part of the search pattern. - bool complex_subs = false; - for (unsigned i=0; i(it->first) || is_exactly_a(it->first)) { + options |= subs_options::pattern_is_product; + break; + } } + if (!(options & subs_options::pattern_is_product)) + options |= subs_options::pattern_is_not_product; + } - if (complex_subs) { + if (options & subs_options::pattern_is_product) { // Substitute in the recombined pairs epvector::const_iterator cit = seq.begin(), last = seq.end(); while (cit != last) { const ex &orig_ex = recombine_pair_to_ex(*cit); - const ex &subsed_ex = orig_ex.subs(ls, lr, no_pattern); + const ex &subsed_ex = orig_ex.subs(m, options); if (!are_ex_trivially_equal(orig_ex, subsed_ex)) { // Something changed, copy seq, subs and return it @@ -1596,7 +1587,7 @@ epvector * expairseq::subschildren(const lst &ls, const lst &lr, bool no_pattern // Copy rest while (cit != last) { - s->push_back(split_ex_to_pair(recombine_pair_to_ex(*cit).subs(ls, lr, no_pattern))); + s->push_back(split_ex_to_pair(recombine_pair_to_ex(*cit).subs(m, options))); ++cit; } return s; @@ -1611,7 +1602,7 @@ epvector * expairseq::subschildren(const lst &ls, const lst &lr, bool no_pattern epvector::const_iterator cit = seq.begin(), last = seq.end(); while (cit != last) { - const ex &subsed_ex = cit->rest.subs(ls, lr, no_pattern); + const ex &subsed_ex = cit->rest.subs(m, options); if (!are_ex_trivially_equal(cit->rest, subsed_ex)) { // Something changed, copy seq, subs and return it @@ -1627,7 +1618,7 @@ epvector * expairseq::subschildren(const lst &ls, const lst &lr, bool no_pattern // Copy rest while (cit != last) { - s->push_back(combine_ex_with_coeff_to_pair(cit->rest.subs(ls, lr, no_pattern), + s->push_back(combine_ex_with_coeff_to_pair(cit->rest.subs(m, options), cit->coeff)); ++cit; }