C ++:boost ptree remove children:没有匹配的功能

时间:2017-07-22 10:32:08

标签: c++ c++11 boost boost-propertytree ptree

在尝试删除boost属性树的子节点时,我使用erase函数中的直接节点导致

error: no matching function for call to 
‘boost::property_tree::basic_ptree<std::__cxx11::basic_string<char>, 
std::__cxx11::basic_string<char> >::erase(std::pair<const 
std::__cxx11::basic_string<char>, 
boost::property_tree::basic_ptree<std::__cxx11::basic_string<char>, 
std::__cxx11::basic_string<char> > >&)’

pt0.erase(pt_child);

代码的正确形式是什么?

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

using namespace boost::property_tree;

void print(const ptree &p)
{
    json_parser::write_json(std::cout, p);
}

int main()
{
    ptree pt0;

    for(int i=0;i<10;i++)
        pt0.put_child("kid"+std::to_string(i+1),ptree());
    print(pt0);

    for(auto& pt_child : pt0)
        pt0.erase(pt_child);
    print(pt0);

    return 0;
}

3 个答案:

答案 0 :(得分:1)

你可以这样做:ptree.get_child(“path.to”)。erase(“child”);请注意,这会删除路径“path.to”中名为“child”的所有节点及其子节点。

答案 1 :(得分:0)

不幸的是,ptree api 没有与 remove_child 一起工作的 property_tree::path 方法,就像 get_child 一样。不过,这里有一个小方法可以做到这一点。重要的部分是您可以使用点表示法来指定要删除的任何子子项。此外,这个实现只删除了一个孩子。

#include <boost/property_tree/ptree.hpp>
#include <boost/algorithm/string.hpp>

// TODO: this should support boost::property_tree::path
// like get_child does to make it obvious that it supports
// the path separator notation for specifying sub children
bool remove_child(boost::property_tree::ptree& pt, const std::string& path) {
  // split up the path into each sub part
  std::vector<std::string> path_parts;
  boost::split(path_parts, path, boost::is_any_of("."));

  // check each part of the path
  auto* root = &pt;
  for (const auto& part : path_parts) {
    // if we dont have this sub child bail
    auto found = root->find(part);
    if (found == root->not_found())
      return false;

    // if this was the last one to look for remove it
    if (&part == &path_parts.back()) {
      root->erase(root->to_iterator(found));
    }// next sub child
    else {
      root = &found->second;
    }
  }

  // made it to the last sub child without bailing on not found
  return true;
}

答案 2 :(得分:-1)

根据docs,您只能.erase key_typeiterator,但您尝试按value_type进行。

您可以这样做

for(auto& [key, data]: pt0)
    pt0.erase(key);

或显式遍历迭代器:

for(auto it = pt0.begin(); it != pt0.end(); ++it)
    pt0.erase(it);

但是,无论如何你要删除所有孩子,更好的方法就是

pt0.clear();