Boostプロパティツリーwrite_jsonを使用してシリアル化しようとしています。すべてを文字列として保存します。データが間違っているわけではありませんが、毎回明示的にキャストする必要があります。 (pythonまたは他のC++ json(non boost)ライブラリのように)
ここにいくつかのサンプルコードとロケールに応じて私が得るものがあります:
boost::property_tree::ptree root, arr, elem1, elem2;
elem1.put<int>("key0", 0);
elem1.put<bool>("key1", true);
elem2.put<float>("key2", 2.2f);
elem2.put<double>("key3", 3.3);
arr.Push_back( std::make_pair("", elem1) );
arr.Push_back( std::make_pair("", elem2) );
root.put_child("path1.path2", arr);
std::stringstream ss;
write_json(ss, root);
std::string my_string_to_send_somewhare_else = ss.str();
およびmy_string_to_send_somewhere_else
はsthです。このような:
{
"path1" :
{
"path2" :
[
{
"key0" : "0",
"key1" : "true"
},
{
"key2" : "2.2",
"key3" : "3.3"
}
]
}
}
とにかく、次のような値として保存する必要があります:"key1" : true
または"key2" : 2.2
?
私が思いついた最もシンプルでクリーンなソリューションは、プレースホルダーを使用してJSONを生成し、最後の文字列で余分な引用符を捨てる実際の値に置き換えることでした。
static string buildGetOrdersCommand() {
ptree root;
ptree element;
element.put<string>("pendingOnly", ":pendingOnly");
element.put<string>("someIntValue", ":someIntValue");
root.put("command", "getOrders");
root.put_child("arguments", element);
std::ostringstream buf;
write_json(buf, root, false);
buf << std::endl;
string json = buf.str();
replace(json, ":pendingOnly", "true");
replace(json, ":someIntValue", std::to_string(15));
return json;
}
static void replace(string& json, const string& placeholder, const string& value) {
boost::replace_all<string>(json, "\"" + placeholder + "\"", value);
}
そして結果は
{「コマンド」:「getOrders」、「引数」:{「pendingOnly」:true、「someIntValue」:15}}
わかりました、私はこのようにそれを解決しました(もちろん、それは少しのハッキングであり、さらなる作業を必要とするので、すべての人に適しているわけではありません)。
独自の_write_json
_関数を作成し(ファイル_json_parser.hpp
_および_json_parser_write.hpp
_をプロジェクトにコピーしただけです)、_json_parser_write.hpp
_の次の行を変更しました。
stream << Ch('"') << data << Ch('"'); ==> stream << data;
次に、文字列を除いて値が適切に保存されるので、カスタムトランスレータを作成しました。
_template <typename T>
struct my_id_translator
{
typedef T internal_type;
typedef T external_type;
boost::optional<T> get_value(const T &v) { return v.substr(1, v.size() - 2) ; }
boost::optional<T> put_value(const T &v) { return '"' + v +'"'; }
};
_
そして、次を使用して単純に保存された文字列:
_elem2.put<std::string>("key2", "asdf", my_id_translator<std::string>());
_
完全なプログラム:
_#include <iostream>
#include <string>
#include <sstream>
#include <boost/property_tree/ptree.hpp>
#include "property_tree/json_parser.hpp" // copied the headers
template <typename T>
struct my_id_translator
{
typedef T internal_type;
typedef T external_type;
boost::optional<T> get_value(const T &v) { return v.substr(1, v.size() - 2) ; }
boost::optional<T> put_value(const T &v) { return '"' + v +'"'; }
};
int main(int, char *[])
{
using namespace std;
using boost::property_tree::ptree;
using boost::property_tree::basic_ptree;
try
{
ptree root, arr,elem2;
basic_ptree<std::string, std::string> elem1;
elem1.put<int>("int", 10 );
elem1.put<bool>("bool", true);
elem2.put<double>("double", 2.2);
elem2.put<std::string>("string", "some string", my_id_translator<std::string>());
arr.Push_back( std::make_pair("", elem1) );
arr.Push_back( std::make_pair("", elem2) );
root.put_child("path1.path2", arr);
std::stringstream ss;
write_json(ss, root);
std::string my_string_to_send_somewhere_else = ss.str();
cout << my_string_to_send_somewhere_else << endl;
}
catch (std::exception & e)
{
cout << e.what();
}
return 0;
}
_
結果:)
_{
"path1":
{
"path2":
[
{
"int": 10,
"bool": true
},
{
"double": 2.2,
"string": "some string"
}
]
}
}
_
Boostは、その実装がJSON標準に100%準拠していないことを確認します。次のリンクを参照して、その説明を確認してください: JSON型を保持するptreeバリアントを作成することは将来の計画ですが、遠い将来です !
Typedef basic_ptree <std :: string、std :: string> ptree;があるので、 boostライブラリでは、boostは常に各値を文字列としてシリアル化し、すべての値を同等の文字列に解析します。
出力されたJSONから、シリアライザーが何らかの.toString()メソッドを使用してすべてを文字列にシリアル化すること、つまり各メンバーのタイプを認識しないため、すべてを ""で囲むことが明らかです。
この問題の詳細については、「 プロパティツリーを使用してBoostでJSON配列を作成する 」を参照してください。
これを解決するために、utilsに別の関数を追加することになりました。
#include <string>
#include <regex>
#include <boost/property_tree/json_parser.hpp>
namespace bpt = boost::property_tree;
typedef bpt::ptree JSON;
namespace boost { namespace property_tree {
inline void write_jsonEx(const std::string & path, const JSON & ptree)
{
std::ostringstream oss;
bpt::write_json(oss, ptree);
std::regex reg("\\\"([0-9]+\\.{0,1}[0-9]*)\\\"");
std::string result = std::regex_replace(oss.str(), reg, "$1");
std::ofstream file;
file.open(path);
file << result;
file.close();
}
} }
お役に立てば幸いです。