повысить исключение сериализации, в и из xml, с помощью SQL server xml datatype
Хорошо, вот моя проблема.
У меня есть объект boost, который создает xml-строку с помощью сериализации, это работает просто отлично.
Использование boost версии 1.38
Я беру эту xml-строку и сохраняю ее в таблице базы данных sql server в xml-тип данных, это также прекрасно работает.
Затем я извлекаю свою xml-строку из таблицы базы данных, но формат немного изменился с момента ее вставки, в основном пустые значения были коротко помечены, от <data></data>
до <data/>
Вот пример xml до и после.
Перед
<grid class_id="0" tracking_level="0" version="0">
<name>test_table</name>
<columns class_id="1" tracking_level="0" version="0">
<count>2</count>
<item_version>0</item_version>
<item class_id="2" tracking_level="1" version="0" object_id="_0">
<label>AAAA</label>
<data>xxxx</data>
</item>
<item class_id_reference="2" object_id="_1">
<label>BBBB</label>
<data></data>
</item>
</columns>
</grid>
После
<grid class_id="0" tracking_level="0" version="0">
<name>test_table</name>
<columns class_id="1" tracking_level="0" version="0">
<count>2</count>
<item_version>0</item_version>
<item class_id="2" tracking_level="1" version="0" object_id="_0">
<label>AAAA</label>
<data>xxxx</data>
</item>
<item class_id_reference="2" object_id="_1">
<label>BBBB</label>
<data /> <!-- NOW SHORT TAGGED -->
</item>
</columns>
</grid>
Это также прекрасно, вполне приемлемо и не является неожиданностью.
Проблема возникает, когда я беру эту xml-строку и пытаюсь сериализовать xml обратно в объекты boost, он выбрасывает и исключение, когда он сталкивается с коротким тегом в xml-строке.
Я наткнулся на кирпичную стену с этим и не знаю, как решить проблему, и не могу найти никаких ссылок на эту проблему в интернете, поэтому любая помощь будет очень признательна.
:)
Вот мой код, он должен компилироваться без каких-либо проблем, вам просто нужно заполнить пробелы для части db:
Сетка.ГЭС
////////////////////////////////////////////////////////////////
// grid boost serialization object
//
#pragma once
#include <string>
#include <iomanip>
#include <iostream>
#include <fstream>
#include <boost/serialization/nvp.hpp>
#include <boost/serialization/utility.hpp>
#include <boost/serialization/list.hpp>
#include <boost/serialization/version.hpp>
/////////////////////////////////////////////////////////////
// Column
//
namespace sdcm
{
class Column
{
public:
// every serializable class needs a constructor
Column()
{
}
Column(const std::string& _label, const std::string& _data)
: label(_label),
data(_data)
{
}
private:
friend class boost::serialization::access;
friend std::ostream & operator<<(std::ostream &os, const Column &col);
std::string label;
std::string data;
template<class Archive>
void serialize(Archive & ar, const unsigned int /* file_version */)
{
ar & BOOST_SERIALIZATION_NVP(label)
& BOOST_SERIALIZATION_NVP(data)
;
}
};
class Grid
{
public:
// every serializable class needs a constructor
Grid()
{
}
Grid(const std::string& _name)
: name(_name)
{
}
void append(Column* col)
{
columns.insert(columns.end(), col);
}
private:
friend class boost::serialization::access;
friend std::ostream & operator<<(std::ostream &os, const Grid &grid);
std::string name;
typedef Column* GRID_COLUMNS;
std::list<GRID_COLUMNS> columns;
template<class Archive>
void serialize(Archive &ar, const unsigned int version)
{
ar & BOOST_SERIALIZATION_NVP(name)
& BOOST_SERIALIZATION_NVP(columns);
}
};
} // end namespace
Главная.cpp
// boost_test.cpp : Defines the entry point for the console application.
//
#include "stdafx.h"
#include <sstream>
#include <iostream>
#include <string>
#include <boost/archive/xml_oarchive.hpp>
#include <boost/archive/xml_iarchive.hpp>
#include "grid.hpp"
std::string get_grid_as_xml_str(sdcm::Grid& grid)
{
std::ostringstream xml ;
unsigned int flags = boost::archive::no_header
//| boost::archive::no_codecvt
//| boost::archive::no_xml_tag_checking
//| boost::archive::no_tracking
;
boost::archive::xml_oarchive oa(xml, flags);
oa << BOOST_SERIALIZATION_NVP(grid);
return xml.str();
}
void restore_grid_from_xml_str(sdcm::Grid& grid, const std::string& xml_str)
{
std::istringstream xml(xml_str);
unsigned int flags = boost::archive::no_header
//| boost::archive::no_codecvt
//| boost::archive::no_xml_tag_checking
//| boost::archive::no_tracking
;
boost::archive::xml_iarchive ia(xml, flags);
ia >> BOOST_SERIALIZATION_NVP(grid);
}
int _tmain(int argc, _TCHAR* argv[])
{
// create grid obj with cols
sdcm::Grid grid("MY GRID");
grid.append(new sdcm::Column("AAAA", "xxxx")) ;
grid.append(new sdcm::Column("BBBB", "")) ;
// get grid as xml string
std::string xml = get_grid_as_xml_str(grid) ;
// Assume xml is saved to SQL Server DB table in XML datatype,
// and the data has be retrived is a shorted tag format used
// where blank values are present in tags
// make a new grid
sdcm::Grid new_grid;
restore_grid_from_xml_str(new_grid, xml);
return 0;
}
1 ответ:
Немного поздно, но я получил ответное письмо от boost.
Да, это своего рода ошибка, но они не собираются ее исправлять.
Вот ответ, если наши заинтересованы:
Код xml_archive, как и весь код сериализации, предполагает, что мы загружаем именно то, что экономим. Вообще стараюсь сделать его больше генерал никогда не казался достойным такого риска. Если вы хотите создать, протестируйте и отправьте патч в spirit parser, который будет обрабатывать теги Я бы посмотрел на него. Но не имея этого, я не склонен тратить время на этом.