⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 ast_xml.cpp

📁 著名的Parser库Spirit在VC6上的Port
💻 CPP
字号:
/*=============================================================================
    ast_xml.hpp

    Spirit V1.2
    URL: http://spirit.sourceforge.net/

    Copyright (c) 2001, Daniel C. Nuffer

    This software is provided 'as-is', without any express or implied
    warranty. In no event will the copyright holder be held liable for
    any damages arising from the use of this software.

    Permission is granted to anyone to use this software for any purpose,
    including commercial applications, and to alter it and redistribute
    it freely, subject to the following restrictions:

    1.  The origin of this software must not be misrepresented; you must
        not claim that you wrote the original software. If you use this
        software in a product, an acknowledgment in the product documentation
        would be appreciated but is not required.

    2.  Altered source versions must be plainly marked as such, and must
        not be misrepresented as being the original software.

    3.  This notice may not be removed or altered from any source
        distribution.
=============================================================================*/

/*
 * This example will parse an XML file into an ast.  Then it will print out
 * the xml using the tree to cout.
 *
 * Note, this example has a flaw. It uses WAY too 
 * much memory, since every single character gets it's own ast node.  A
 * good solution would be to create a parser and put it onto certain points
 * of the xml grammar (like Name, EntityValue, AttValue, SystemLiteral, etc.)
 * that would in essence create a new terminal node and discard the tree
 * it received.  Thus the terminal nodes would be an indentifier instead of
 * a single char.
 */


//#define SPIRIT_DEBUG  ///$$$ DEFINE THIS WHEN DEBUGGING $$$///

#include <boost/spirit/spirit.hpp>
#include <boost/multi_pass.hpp>

#include <vector>
#include <iostream>
#include <fstream>

#include "xml_grammar.hpp"

using namespace spirit;
using namespace std;

/* This is provided so that we can *see* the debug output */
ostream& operator<<(ostream& out, wchar_t c)
{
    out << char(c);
    return out;
}

ostream& operator<<(ostream& out, wchar_t const* c)
{
    while (*c)
        out << char(*c++);
    return out;
}

typedef vector<wchar_t>::iterator iterator_t;
typedef ast_node<iterator_t> node_t;
typedef tree_node_ch_iter<node_t> node_iter_t;
typedef ast_match<iterator_t> ast_match_t;
typedef ast_match_traits<iterator_t> match_traits_t;

void print_tree(ast_match_t hit);
void print_node(node_iter_t const& i);

template <typename RuleT>
static void ParseFile(RuleT const& rule, char const* filename)
{
    cout << "////////////// Parsing with std::vector /////////////////\n\n";

    ifstream in(filename);
    if (!in)
    {
        cout << "Could not open file: " << filename << endl;
        return;
    }

    // Turn of white space skipping on the stream
    in.unsetf(ios::skipws);

    vector<unsigned char> temp = vector<unsigned char>(
            istream_iterator<unsigned char>(in),
            istream_iterator<unsigned char>());

    // This is a rather pathetic attempt to work around gcc 2.95.x's lacking
    // library, which doesn't have wifstream.  The correct thing to do here
    // would be to translate the file into wide characters, but for now it
    // works just fine with ASCII files.
    vector<wchar_t> data = vector<wchar_t>(temp.begin(),
            temp.end());

    iterator_t first = data.begin();
    iterator_t last = data.end();
    ast_match_t hit = rule.ast_parse(first, last, match_traits_t());

    if (hit && first == last)
    {
        cout << "\t\t" << filename << " Parses OK\n\n\n";

        print_tree(hit);
    }
    else
    {
        cout << "\t\t" << filename << " Fails Parsing\n" << endl;
        for (int i = 0; i < 50; ++i)
        {
            if (first == last)
                break;
            cout << char(*first++);
        }
        cout << endl;
    }
}

void print_tree(ast_match_t hit)
{
    print_node(hit.ast.root().ch_begin());
}

void print_node(node_iter_t const& i)
{
    if (i->ch_begin() == i->ch_end()) // terminal node
    {
        std::copy(i->value.first, i->value.last, ostream_iterator<char>(std::cout));
    }
    else
    {
        for(node_iter_t chi = i->ch_begin();
                chi != i->ch_end(); ++chi)
        {
            print_node(chi);
        }

    }
}


int main(int argc, char* argv[])
{
    xml_grammar g;
    if (argc == 2)
    {
        ParseFile(g, argv[1]);
    }
    else
    {
        cout << "Error: no filename given\n";
        return 1;
    }

    return 0;
}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -