Boost :: property_tree: использование std :: vector & lt; & gt; в XML-парсере для хранения нескольких значений в одном ключе

Мой вопрос связан с этим вопросом: Boost property_tree: несколько значений на ключ и на вопрос, следующий за этим вопросом: Boost property_tree: несколько значений на ключ в классе шаблона.

Я пытаюсь проанализировать файл XML, в котором несколько значений перечислены в одном значении ключа с помощью std::vector<>, Следующий код — это то, что я реализовал до сих пор:

#include <boost/optional.hpp>
#include <boost/property_tree/xml_parser.hpp>

namespace boost { namespace property_tree
{

template<typename type>
struct vector_xml_translator
{
boost::optional<std::vector<type> > get_value(const std::string& str)
{
if (!str.empty())
{
std::vector<type> values;
std::stringstream ss(str);

while (ss)
{
type temp_value;
ss >> temp_value;
values.push_back(temp_value);
}

return boost::optional<std::vector<type> >(values);
}
else
{
return boost::optional<std::vector<type> >(boost::none);
}
}

boost::optional<std::string> put_value(const std::vector<type>& b)
{
std::stringstream ss;
for (unsigned int i = 0; i < b.size(); i++)
{
ss << b[i];
if (i != b.size()-1)
{
ss << " ";
}
}
return boost::optional<std::string>(ss.str());
}
};

template<typename ch, typename traits, typename alloc, typename data_type>
struct translator_between<std::basic_string<ch, traits, alloc>, std::vector<data_type> >
{
typedef vector_xml_translator<data_type> type;
};

} // namespace property_tree
} // namespace boost

Минимальный пример для проверки этого кода выглядит следующим образом:

#include <fstream>
#include <iostream>
#include <boost/property_tree/ptree.hpp>
#include <XML_Vector_Translator.hpp>int main()
{
using boost::property_tree::ptree;

std::vector<double> test_vector;

test_vector.push_back(1);
test_vector.push_back(6);
test_vector.push_back(3);ptree pt;
pt.add("base", test_vector);

std::ofstream os("test_file.xml");

write_xml(os, pt, boost::property_tree::xml_writer_settings<std::string>(' ', 2));

std::ifstream is("test_file.xml");

ptree pt_2;
read_xml(is, pt_2);

std::vector<int> test_vector_2;

test_vector_2 = pt_2.get<std::vector<int> >("base");

for (unsigned int i = 0; i < test_vector_2.size(); i++)
{
std::cout << test_vector_2[i] << std::endl;
}

return 0;
}

Когда я запускаю этот код, я получаю ряд ошибок, из-за которых я полагаю, что регистрация структуры переводчика неверна. У кого-нибудь есть идеи, как решить эту проблему и / или улучшить этот код?

2

Решение

  1. Как старый ответ также указывает, что вы должны удовлетворять требованиям для boost::property_tree::detail::is_translatorтак что вам нужно internal_type / external_type Определения типов.

    typedef T internal_type;
    typedef T external_type;
    
  2. Далее, цикл неправильный, вам нужно проверить результат извлечения значения:

    while (ss >> temp_value)
    values.push_back(temp_value);
    
  3. Это плохая практика — помещать свои собственные типы в пространство имен Boost. Только специализация translator_between<> должен быть там.

  4. Вы можете упростить и обобщить много кода

Все в рабочем демо:

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

#include <boost/optional.hpp>
#include <boost/property_tree/ptree.hpp>
#include <vector>
#include <list>

namespace mylib { namespace xml_translators {

template<typename T> struct container
{
// types
typedef T internal_type;
typedef T external_type;

boost::optional<T> get_value(const std::string& str) const
{
if (str.empty())
return boost::none;

T values;
std::stringstream ss(str);

typename T::value_type temp_value;
while (ss >> temp_value)
values.insert(values.end(), temp_value);

return boost::make_optional(values);
}

boost::optional<std::string> put_value(const T& b) {
std::stringstream ss;
size_t i = 0;
for (auto v : b)
ss << (i++?" ":"") << v;
return ss.str();
}
};

} }

namespace boost { namespace property_tree {
template<typename ch, typename traits, typename alloc, typename T>
struct translator_between<std::basic_string<ch, traits, alloc>, std::vector<T> > {
typedef mylib::xml_translators::container<std::vector<T> > type;
};

template<typename ch, typename traits, typename alloc, typename T>
struct translator_between<std::basic_string<ch, traits, alloc>, std::list<T> > {
typedef mylib::xml_translators::container<std::list<T> > type;
};
} }

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

int main()
{
std::stringstream ss;
using boost::property_tree::ptree;

{
ptree pt;
pt.add("base", std::vector<double> { 1, 6, 3 });

write_xml(ss, pt, boost::property_tree::xml_writer_settings<std::string>(' ', 2));
}

{
ptree pt;
read_xml(ss, pt);

std::cout << "As string: '" << pt.get("base", "") << "'\n";
auto roundtrip = pt.get<std::list<int> >("base");
for (auto i : roundtrip)
std::cout << i << std::endl;
}
}

Печать

As string: '1 6 3'
1
6
3

¹ Boost property_tree: несколько значений на ключ, см. также переводчик http://www.boost.org/doc/libs/1_64_0/doc/html/boost/property_tree/id_translator.html

1

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

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

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