6

我正在使用 boost 属性树迭代 XML 文档并将结果存储在结构中。我遇到的问题是我只能到达第一个“项目”节点而无法访问第二个“项目”节点。我希望有人能指出我在哪里犯了错误。

我的程序输出如下所示(您可以看到缺少项目。没有显示 cookie2、candy2 或 Chocolate2 项目):

jar : snAcks
snack : coOkie
item : cooKie1
snack : canDy
item : caNdy1
snack : cHocolate
item : choColate1

这是xml文件:

<root>
    <jar name="snAcks">
        <snack name="coOkie">
           <item name="cooKie1"></item>
           <item name="cookIe2"></item>
        </snack>
        <snack name="canDy">
           <item name="caNdy1"></item>
           <item name="candY2"></item>
        </snack>
        <snack name="cHocolate">
           <item name="choColate1"></item>
           <item name="chocOlate2"></item>
        </snack>
    </jar>
</root>

这是源代码:

void parse_xml( boost::property_tree::iptree const& pt )
{
    BOOST_FOREACH( boost::property_tree::iptree::value_type const& v, pt.get_child("root.jar") )
    {
        // Show jar
        if ( boost::iequals( v.first, "<xmlattr>" ) )
        {
            std::cout << "jar : " << v.second.get<std::string>("NaME") << std::endl;
        }

        if ( boost::iequals( v.first, "snack" ) )
        {
            // Show snack
            std::cout << "snack : " << v.second.get<std::string>("<xmlattr>.name")  << std::endl;

            try
            {
                BOOST_FOREACH( boost::property_tree::iptree::value_type const& val, v.second.get_child("item") )
                {
                    if ( boost::iequals( val.first, "<xmlattr>" ) ) {
                        // Show item
                        std::cout << "item : " << val.second.get<std::string>("nAmE")  << std::endl;
                    }
                }
            }

            catch (boost::property_tree::ptree_bad_path& e)
            {
                // Show what caused exception
                std::cout << "Exception: " << e.what() << std::endl;
            }
        }
    }
}

感谢您花时间阅读本文。我想我犯了一个简单的错误,但不明白在哪里。

4

1 回答 1

11

我想通了,但这不是我所说的直观的 xml 解析器库。

void parse_xml( boost::property_tree::iptree const& pt )
{
    BOOST_FOREACH(boost::property_tree::iptree::value_type const& v, pt.get_child("root.jar"))
    {
        // Show jar
        if ( boost::iequals( v.first, "<xmlattr>" ) ) {
            std::cout << "jar : " << v.second.get<std::string>("NaME") << std::endl;
        }

        if ( boost::iequals( v.first, "snack" ) )
        {
            BOOST_FOREACH(boost::property_tree::iptree::value_type const& val, v.second)
            {
                // Show snack
                if ( boost::iequals( val.first, "<xmlattr>" ) ) {
                    std::cout << "snack : " << val.second.get<std::string>("name")  << std::endl;
                }

                if ( boost::iequals(val.first, "item") )
                {
                    BOOST_FOREACH(boost::property_tree::iptree::value_type const& val2, val.second)
                    if ( boost::iequals( val2.first, "<xmlattr>" ) ) {
                        // Show item
                        std::cout << "item : " << val2.second.get<std::string>("nAmE")  << std::endl;
                    }
                }
            }
        }
    }
}

如果你比较这两个代码片段,你会发现我的结构更规则一些。

  1. 遍历所有节点
  2. 处理<xmlattr>节点
  3. process the "real" node.
  4. repeat from step 1. inside that node.
于 2012-03-02T20:28:58.560 回答