XML-путь ко второму дочернему элементу

Я хочу поместить новое значение для дочернего элемента в дереве через путь (без итерации)

Как пример:

#include <iostream>
#include <boost/property_tree/ptree.hpp>
#include <boost/property_tree/xml_parser.hpp>

int main() {
using boost::property_tree::ptree;

ptree main;
ptree children;
children.add("Foo", "bar");
main.add_child("child", children);
main.add_child("child", children);

ptree newChildren("Foo");
main.put("child{2}.Foo", newChildren.data()); // <-- Access second element?

// Output
boost::property_tree::xml_writer_settings<std::string> settings('\t', 1);
boost::property_tree::write_xml(std::cout, main, settings);

std::cin.ignore();
return 0;
}

Проблема в том, что я не могу получить доступ ко второму ребенку по пути. Есть ли форматирование, которое работает?

Я имею в виду эту {2} часть:

"child{2}.Foo"

Я старался <2>, [2], (2) … не повезло … 🙁

Есть надежда на меня? Спасибо!

4

Решение

Повторюсь: в Boost нет парсера / библиотеки xml.

То, что вы используете (ab), это Boost Property Tree. Как вы обнаружили, это библиотека «дерева свойств», то есть она может делать некоторые вещи. Включая запись и чтение деревьев свойств.

Если вы хотите использовать XML общего назначения, подумайте об использовании библиотеки XML (Какой анализатор XML я должен использовать в C ++?).

Без дальнейших церемоний:

Жить на Колиру

#include <iostream>
#include <boost/property_tree/ptree.hpp>
#include <boost/property_tree/xml_parser.hpp>

int main() {
using boost::property_tree::ptree;

ptree pt;

{
ptree children; children.add("Foo", "bar");
pt.add_child("child", children);
pt.add_child("child", children).put("MARK","so we know it's the second");
}

// Output
boost::property_tree::xml_writer_settings<std::string> settings('\t', 1);
//boost::property_tree::write_xml(std::cout, pt, settings);

auto child1 = std::find_if(pt.begin(),        pt.end(), [](auto& node) { return node.first == "child"; });
auto child2 = std::find_if(std::next(child1), pt.end(), [](auto& node) { return node.first == "child"; });

if (child2 != pt.end())
{
boost::property_tree::write_xml(std::cout, child2->second, settings);

ptree newChildren("Foo");
child2->second.put("Sub.Foo", newChildren.data()).put("BYE", "ALL DONE"); // <-- Access second element?
boost::property_tree::write_xml(std::cout << "\n\nAFTER EDITING:\n", child2->second, settings);
}
}

Печать:

<?xml version="1.0" encoding="utf-8"?>
<Foo>bar</Foo>
<MARK>so we know it&apos;s the second</MARK>AFTER EDITING:
<?xml version="1.0" encoding="utf-8"?>
<Foo>bar</Foo>
<MARK>so we know it&apos;s the second</MARK>
<Sub>
<Foo>
Foo
<BYE>ALL DONE</BYE>
</Foo>
</Sub>

ОБНОВИТЬ

Улучшение стиля с помощью вспомогательной функции:

Жить на Колиру

#include <iostream>
#include <boost/property_tree/ptree.hpp>
#include <boost/property_tree/xml_parser.hpp>

template <typename Tree, typename Out>
Out find_all(Tree& pt, typename Tree::path_type path, Out out) {
if (path.empty()) {
*out++ = pt;
return out;
}

auto head = path.reduce();
for (auto& child : pt)
if (child.first == head)
out = find_all(child.second, path, out);

return out;
}

int main() {
using boost::property_tree::ptree;

ptree pt;

{
ptree children; children.add("Foo", "bar");
pt.add_child("child", children);
pt.add_child("child", children).put("MARK","so we know it's the second");
}

// Output
boost::property_tree::xml_writer_settings<std::string> settings('\t', 1);
//boost::property_tree::write_xml(std::cout, pt, settings);

std::vector<std::reference_wrapper<ptree> > matches;
find_all(pt, "child", back_inserter(matches));

ptree& child2 = matches.at(1);
child2.put("BYE", "ALL DONE");

boost::property_tree::write_xml(std::cout, child2, settings);
}

Печать

<?xml version="1.0" encoding="utf-8"?>
<Foo>bar</Foo>
<MARK>so we know it&apos;s the second</MARK>
<BYE>ALL DONE</BYE>
3

Другие решения

Других решений пока нет …

По вопросам рекламы [email protected]