为什么提升属性树write_json将所有内容保存为字符串?有可能改变吗?

ppr*_*mek 61 json boost boost-propertytree

我想使用boost属性树write_json序列化,这样可以节省一切为字符串,这并不是说数据是错误的,但我需要每次都明确地投下他们,我想别的地方使用它们.(比如在python或其他C++ json(非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();
Run Code Online (Sandbox Code Playgroud)

并且my_string_to_send_somewhere_else是某事.像这样:

{
    "path1" :
    {
       "path2" :
       [
            {
                 "key0" : "0",
                 "key1" : "true"
            },
            {
                 "key2" : "2.2",
                 "key3" : "3.3"
            }
       ]
    }
}
Run Code Online (Sandbox Code Playgroud)

反正是将它们保存为值,例如: "key1" : true"key2" : 2.2

ppr*_*mek 34

好吧,我已经解决了这个问题,(当然它不适合所有人,因为它有点像黑客,需要进一步的工作).


我已经编写了自己的write_json函数(只需复制文件json_parser.hppjson_parser_write.hpp我的项目)并修改了以下几行json_parser_write.hpp:

  1. 评论第37行 - 逃避引用'''
  2. 更改第76行 - 这样它就不再添加引号: 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 +'"'; }
};
Run Code Online (Sandbox Code Playgroud)

并使用以下方法保存字符串

elem2.put<std::string>("key2", "asdf", my_id_translator<std::string>());
Run Code Online (Sandbox Code Playgroud)

完整的程序:

#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;
}
Run Code Online (Sandbox Code Playgroud)

结果:)

{
    "path1":
    {
        "path2":
        [
            {
                "int": 10,
                "bool": true
            },
            {
                "double": 2.2,
                "string": "some string"
            }
        ]
    }
}
Run Code Online (Sandbox Code Playgroud)

  • +1很有用,知道现在没有干净的解决方案,并且必须使用kludge (5认同)

Ale*_*Sed 12

Boost确认其实施与JSON标准没有100%的一致性.检查以下链接以查看其解释: 制作保留JSON类型的ptree变体是未来的计划,但远远不够.!


小智 6

我能想到的最简单,最干净的解决方案是使用占位符生成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);
}
Run Code Online (Sandbox Code Playgroud)

结果是

{“ command”:“ getOrders”,“ arguments”:{“ pendingOnly”:true,“ someIntValue”:15}}


小智 5

我最终向我的实用程序添加了另一个函数来解决这个问题:

#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();
    }
} }
Run Code Online (Sandbox Code Playgroud)

希望有帮助。