如何在迭代时从poco json中删除密钥?

时间:2019-01-31 06:10:39

标签: c++ poco-libraries

如何在迭代时从Poco json中删除密钥?喜欢:

Poco::JSON::Object::Ptr poco_json;
for (auto& objs : *poco_json)
{
     // do something
     if (objs.first == "specific key")
         poco_json->remove(key);     
}

Poco::JSON::Object::Ptr poco_json;
for(auto it = poco_json->begin();it != poco_json->end();)
{
    // do something
    if (it->first == "specific key")
        it = poco_json->remove(it->first);//error : poco didn't have something like this
    else
        ++it;
}

问题是从json中删除密钥后,它将使迭代器无效。我知道在std :: map中,擦除会为下一次迭代返回有效的迭代器,但是我找不到与Poco json类似的东西。

2 个答案:

答案 0 :(得分:1)

std::map::erase返回迭代器的下一个项目,因为C ++ 11,以这种方式删除项C ++ 11之前:

for (auto it = m.begin(); it != m.end(); ) {
    if (it->first == someKey)
      m.erase(it++); // use post-increment,pass copy of iterator, advance it
    else 
      ++it;
  }

,您可以用类似的方式删除Poco::JSON::Object中的密钥。您在哪里读到remove使迭代器无效?

从源一些片断代码:

class JSON_API Object {
    typedef std::map<std::string, Dynamic::Var> ValueMap; // <--- map
    //...
    Iterator begin();
        /// Returns begin iterator for values.
    Iterator end();
        /// Returns end iterator for values.
    void remove(const std::string& key);
        /// Removes the property with the given key.

    ValueMap          _values; // <---

};

inline Object::Iterator Object::begin()
{
    return _values.begin();
}
inline Object::Iterator Object::end()
{
    return _values.end();
}

inline void Object::remove(const std::string& key)
{
    _values.erase(key); // <--- erase is called on map, so iteratos are not invalidated
    if (_preserveInsOrder)
    {
        KeyList::iterator it = _keys.begin();
        KeyList::iterator end = _keys.end();
        for (; it != end; ++it)
        {
            if (key == (*it)->first)
            {
                _keys.erase(it);
                break;
            }
        }
    }
    _modified = true;
}

您可以重写你的循环变成:

for(auto it = poco_json->begin();it != poco_json->end();)
{
    // do something
    if (it->first == "specific key")
    {
        auto copyIt = it++;
        poco_json->remove(copyIt->first);
    }
    else
        ++it;
}

编辑 为什么你的代码不会对工作范围-for循环:

for (auto& objs : *poco_json)
{
     // do something
     if (objs.first == "specific key")
         poco_json->remove(key);     
}

它被翻译成

for (auto it = poco_json->begin(); it != poco_json->end(); ++it)
{
     // do something
     if (it->first == "specific key")
         poco_json->remove(it->first);     
    // remove is called, it is erased from inner map
    // ++it is called on iterator which was invalidated, 
    // code crashes
}

答案 1 :(得分:0)

您可以在Poco中修改此代码:

inline Iterator Object::remove(const std::string& key)
{
    auto ret_it = _values.erase(key);
    if (_preserveInsOrder)
    {
        KeyList::iterator it = _keys.begin();
        KeyList::iterator end = _keys.end();
        for (; it != end; ++it)
        {
            if (key == (*it)->first)
            {
                _keys.erase(it);
                break;
            }
        }
    }
    _modified = true;

  return ret_it;
}