Мой вопрос связан с этим вопросом: 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;
}
Когда я запускаю этот код, я получаю ряд ошибок, из-за которых я полагаю, что регистрация структуры переводчика неверна. У кого-нибудь есть идеи, как решить эту проблему и / или улучшить этот код?
Как старый ответ также указывает, что вы должны удовлетворять требованиям для boost::property_tree::detail::is_translator
так что вам нужно internal_type
/ external_type
Определения типов.
typedef T internal_type;
typedef T external_type;
Далее, цикл неправильный, вам нужно проверить результат извлечения значения:
while (ss >> temp_value)
values.push_back(temp_value);
Это плохая практика — помещать свои собственные типы в пространство имен Boost. Только специализация translator_between<>
должен быть там.
Вы можете упростить и обобщить много кода
Все в рабочем демо:
#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
Других решений пока нет …