void test_depth_first_iterator::test_dfi() { std::string infix("7-(3+(4-6))+1*8"); CEvaluationTree* pTree = new CEvaluationTree(); pTree->setInfix(infix); std::string s = pTree->getInfix(); CPPUNIT_ASSERT(pTree->getRoot() != NULL); CEvaluationNodeDepthFirstIterator it(pTree->getRoot()); CPPUNIT_ASSERT(it.isValid()); CEvaluationNode* pNode = NULL; CEvaluationNodeNumber* pNumberNode = NULL; CEvaluationNodeOperator* pOperatorNode = NULL; // 7 pNode = *it; CPPUNIT_ASSERT(pNode != NULL); CPPUNIT_ASSERT(CEvaluationNode::type(pNode->getType()) == CEvaluationNode::NUMBER); pNumberNode = dynamic_cast<CEvaluationNodeNumber*>(pNode); CPPUNIT_ASSERT(pNumberNode != NULL); CPPUNIT_ASSERT((fabs(pNumberNode->getValue() - 7) / 7.0) <= std::numeric_limits<double>::min()); // 3 ++it; CPPUNIT_ASSERT(it.isValid() == true); pNode = *it; CPPUNIT_ASSERT(pNode != NULL); CPPUNIT_ASSERT(CEvaluationNode::type(pNode->getType()) == CEvaluationNode::NUMBER); pNumberNode = dynamic_cast<CEvaluationNodeNumber*>(pNode); CPPUNIT_ASSERT(pNumberNode != NULL); CPPUNIT_ASSERT((fabs(pNumberNode->getValue() - 3) / 3.0) <= std::numeric_limits<double>::min()); // 4 ++it; CPPUNIT_ASSERT(it.isValid() == true); pNode = *it; CPPUNIT_ASSERT(pNode != NULL); CPPUNIT_ASSERT(CEvaluationNode::type(pNode->getType()) == CEvaluationNode::NUMBER); pNumberNode = dynamic_cast<CEvaluationNodeNumber*>(pNode); CPPUNIT_ASSERT(pNumberNode != NULL); CPPUNIT_ASSERT((fabs(pNumberNode->getValue() - 4) / 4.0) <= std::numeric_limits<double>::min()); // 6 ++it; CPPUNIT_ASSERT(it.isValid() == true); pNode = *it; CPPUNIT_ASSERT(pNode != NULL); CPPUNIT_ASSERT(CEvaluationNode::type(pNode->getType()) == CEvaluationNode::NUMBER); pNumberNode = dynamic_cast<CEvaluationNodeNumber*>(pNode); CPPUNIT_ASSERT(pNumberNode != NULL); CPPUNIT_ASSERT((fabs(pNumberNode->getValue() - 6) / 6.0) <= std::numeric_limits<double>::min()); // - ++it; CPPUNIT_ASSERT(it.isValid() == true); pNode = *it; CPPUNIT_ASSERT(pNode != NULL); CPPUNIT_ASSERT(CEvaluationNode::type(pNode->getType()) == CEvaluationNode::OPERATOR); pOperatorNode = dynamic_cast<CEvaluationNodeOperator*>(pNode); CPPUNIT_ASSERT(pOperatorNode != NULL); CPPUNIT_ASSERT((CEvaluationNodeOperator::SubType)CEvaluationNode::subType(pOperatorNode->getType()) == CEvaluationNodeOperator::MINUS); // + ++it; CPPUNIT_ASSERT(it.isValid() == true); pNode = *it; CPPUNIT_ASSERT(pNode != NULL); CPPUNIT_ASSERT(CEvaluationNode::type(pNode->getType()) == CEvaluationNode::OPERATOR); pOperatorNode = dynamic_cast<CEvaluationNodeOperator*>(pNode); CPPUNIT_ASSERT(pOperatorNode != NULL); CPPUNIT_ASSERT((CEvaluationNodeOperator::SubType)CEvaluationNode::subType(pOperatorNode->getType()) == CEvaluationNodeOperator::PLUS); // - ++it; CPPUNIT_ASSERT(it.isValid() == true); pNode = *it; CPPUNIT_ASSERT(pNode != NULL); CPPUNIT_ASSERT(CEvaluationNode::type(pNode->getType()) == CEvaluationNode::OPERATOR); pOperatorNode = dynamic_cast<CEvaluationNodeOperator*>(pNode); CPPUNIT_ASSERT(pOperatorNode != NULL); CPPUNIT_ASSERT((CEvaluationNodeOperator::SubType)CEvaluationNode::subType(pOperatorNode->getType()) == CEvaluationNodeOperator::MINUS); // 1 ++it; CPPUNIT_ASSERT(it.isValid() == true); pNode = *it; CPPUNIT_ASSERT(pNode != NULL); CPPUNIT_ASSERT(CEvaluationNode::type(pNode->getType()) == CEvaluationNode::NUMBER); pNumberNode = dynamic_cast<CEvaluationNodeNumber*>(pNode); CPPUNIT_ASSERT(pNumberNode != NULL); CPPUNIT_ASSERT((fabs(pNumberNode->getValue() - 1) / 1.0) <= std::numeric_limits<double>::min()); // 8 ++it; CPPUNIT_ASSERT(it.isValid() == true); pNode = *it; CPPUNIT_ASSERT(pNode != NULL); CPPUNIT_ASSERT(CEvaluationNode::type(pNode->getType()) == CEvaluationNode::NUMBER); pNumberNode = dynamic_cast<CEvaluationNodeNumber*>(pNode); CPPUNIT_ASSERT(pNumberNode != NULL); CPPUNIT_ASSERT((fabs(pNumberNode->getValue() - 8) / 8.0) <= std::numeric_limits<double>::min()); // * ++it; CPPUNIT_ASSERT(it.isValid() == true); pNode = *it; CPPUNIT_ASSERT(pNode != NULL); CPPUNIT_ASSERT(CEvaluationNode::type(pNode->getType()) == CEvaluationNode::OPERATOR); pOperatorNode = dynamic_cast<CEvaluationNodeOperator*>(pNode); CPPUNIT_ASSERT(pOperatorNode != NULL); CPPUNIT_ASSERT((CEvaluationNodeOperator::SubType)CEvaluationNode::subType(pOperatorNode->getType()) == CEvaluationNodeOperator::MULTIPLY); // + ++it; CPPUNIT_ASSERT(it.isValid() == true); pNode = *it; CPPUNIT_ASSERT(pNode != NULL); CPPUNIT_ASSERT(CEvaluationNode::type(pNode->getType()) == CEvaluationNode::OPERATOR); pOperatorNode = dynamic_cast<CEvaluationNodeOperator*>(pNode); CPPUNIT_ASSERT(pOperatorNode != NULL); CPPUNIT_ASSERT((CEvaluationNodeOperator::SubType)CEvaluationNode::subType(pOperatorNode->getType()) == CEvaluationNodeOperator::PLUS); ++it; CPPUNIT_ASSERT(it.isValid() == false); CPPUNIT_ASSERT(*it == NULL); delete pTree; }
bool CEvaluationNodeNormalizer::collectIdenticalBranches(CEvaluationNodeOperator::SubType subType, std::vector<CEvaluationNode*>& chainNodes) { bool changed = false; if (chainNodes.size() > 1) { // what is done depends on the subtype // if it is an additon we replace identical nodes with a multiplication of // the node with the number of occurences // for a multiplication it's the same, but we use a power instead of a // multiplication if (subType == CEvaluationNodeOperator::MULTIPLY || subType == CEvaluationNodeOperator::PLUS) { CEvaluationNodeOperator* pOperation = (subType == CEvaluationNodeOperator::MULTIPLY) ? new CEvaluationNodeOperator(CEvaluationNodeOperator::POWER, "^") : new CEvaluationNodeOperator(CEvaluationNodeOperator::MULTIPLY, "*"); std::vector<CEvaluationNode*>::iterator it = chainNodes.begin(), endit = chainNodes.end(); std::map<CEvaluationNode, unsigned int> occurenceMap; while (it != endit) { std::map<CEvaluationNode, unsigned int>::iterator pos = occurenceMap.find(**it); if (pos == occurenceMap.end()) { occurenceMap[**it] = 1; } else { ++(pos->second); changed = true; } // delete the node delete(*it); ++it; } chainNodes.clear(); // convert the contents of the occurenceMap to new nodes std::map<CEvaluationNode, unsigned int>::const_iterator mapIt = occurenceMap.begin(), mapEndit = occurenceMap.end(); std::ostringstream os; while (mapIt != mapEndit) { os << mapIt->second; CEvaluationNodeNumber* pNumberNode = new CEvaluationNodeNumber(CEvaluationNodeNumber::INTEGER, os.str()); CEvaluationNodeOperator* pNewOperator = dynamic_cast<CEvaluationNodeOperator*>(CEvaluationNode::create(pOperation->getType(), pOperation->getData())); pNewOperator->addChild(pNumberNode); pNewOperator->addChild(mapIt->first.copyBranch()); chainNodes.push_back(pNewOperator); os.str(""); ++mapIt; } delete pOperation; } } return changed; }