// $Id: SyntaxTree.cpp 1960 2008-12-15 12:52:38Z phkoehn $ // vim:tabstop=2 /*********************************************************************** Moses - factored phrase-based language decoder Copyright (C) 2009 University of Edinburgh This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2.1 of the License, or (at your option) any later version. This library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this library; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA ***********************************************************************/ #include "SyntaxTree.h" #include #include SyntaxTree::~SyntaxTree() { Clear(); } void SyntaxTree::Clear() { m_top = 0; // loop through all m_nodes, delete them for(size_t i=0; i= 2 for( int length=2; length<=size; length++ ) { for( int startPos = 0; startPos <= size-length; startPos++ ) { if (HasNode( startPos, startPos+length-1 )) { // processing one (parent) span //std::cerr << "# " << startPos << "-" << (startPos+length-1) << ":"; SplitPoints splitPoints; splitPoints.push_back( startPos ); //std::cerr << " " << startPos; int first = 1; int covered = 0; while( covered < length ) { // find largest covering subspan (child) // starting at last covered position for( int midPos=length-first; midPos>covered; midPos-- ) { if( HasNode( startPos+covered, startPos+midPos-1 ) ) { covered = midPos; splitPoints.push_back( startPos+covered ); // std::cerr << " " << ( startPos+covered ); first = 0; } } } // std::cerr << std::endl; parents.push_back( splitPoints ); } } } return parents; } bool SyntaxTree::HasNode( int startPos, int endPos ) const { return GetNodes( startPos, endPos).size() > 0; } const std::vector< SyntaxNode* >& SyntaxTree::GetNodes( int startPos, int endPos ) const { SyntaxTreeIndexIterator startIndex = m_index.find( startPos ); if (startIndex == m_index.end() ) return m_emptyNode; SyntaxTreeIndexIterator2 endIndex = startIndex->second.find( endPos ); if (endIndex == startIndex->second.end()) return m_emptyNode; return endIndex->second; } // for printing out tree std::string SyntaxTree::ToString() const { std::stringstream out; out << *this; return out.str(); } void SyntaxTree::ConnectNodes() { typedef SyntaxTreeIndex2::const_reverse_iterator InnerIterator; SyntaxNode *prev = 0; // Iterate over all start indices from lowest to highest. for (SyntaxTreeIndexIterator p = m_index.begin(); p != m_index.end(); ++p) { const SyntaxTreeIndex2 &inner = p->second; // Iterate over all end indices from highest to lowest. for (InnerIterator q = inner.rbegin(); q != inner.rend(); ++q) { const std::vector &nodes = q->second; // Iterate over all nodes that cover the same span in order of tree // depth, top-most first. for (std::vector::const_reverse_iterator r = nodes.rbegin(); r != nodes.rend(); ++r) { SyntaxNode *node = *r; if (!prev) { // node is the root. m_top = node; node->SetParent(0); } else if (prev->GetStart() == node->GetStart()) { // prev is the parent of node. assert(prev->GetEnd() >= node->GetEnd()); node->SetParent(prev); prev->AddChild(node); } else { // prev is a descendant of node's parent. The lowest common // ancestor of prev and node will be node's parent. SyntaxNode *ancestor = prev->GetParent(); while (ancestor->GetEnd() < node->GetEnd()) { ancestor = ancestor->GetParent(); } assert(ancestor); node->SetParent(ancestor); ancestor->AddChild(node); } prev = node; } } } } std::ostream& operator<<(std::ostream& os, const SyntaxTree& t) { size_t size = t.m_index.size(); for(size_t length=1; length<=size; length++) { for(size_t space=0; spaceGetLabel() + "#######"; os << label.substr(0,7) << " "; } else { os << "------- "; } } os << std::endl; } return os; }