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

时间:2010-05-18 08:52:56

标签: 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();

my_string_to_send_somewhere_else是......像这样:

{
    "path1" :
    {
       "path2" :
       [
            {
                 "key0" : "0",
                 "key1" : "true"
            },
            {
                 "key2" : "2.2",
                 "key3" : "3.3"
            }
       ]
    }
}

无论如何都要将它们保存为值,例如: "key1" : true"key2" : 2.2

7 个答案:

答案 0 :(得分:33)

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


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

  1. 评论第37行 - 逃避引用'''
  2. 更改了第76行 - 因此它不再添加引号: stream << Ch('"') << data << Ch('"'); ==> stream << data;

  3. 然后除了字符串之外,值将被正确保存,所以我为它编写了自定义翻译器:

    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"
                }
            ]
        }
    }
    

答案 1 :(得分:10)

Boost确认其实施与JSON标准没有100%的一致性。请查看以下链接以查看其说明: Making a ptree variant that preserves JSON types is a future plan, but far off.

答案 2 :(得分:5)

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

结果是

  

{&#34;命令&#34;:&#34; getOrders&#34;&#34;参数&#34; {&#34; pendingOnly&#34;:真,&#34; someIntValue&#34 ;:15}}

答案 3 :(得分:3)

因为我们有typedef basic_ptree&lt; std :: string,std :: string&gt; ptree中;在boost库中,boost将始终将每个值序列化为字符串,并将所有值解析为等效的字符串。

答案 4 :(得分:2)

从输出的JSON中可以看出,序列化程序使用某种.toString()方法将所有内容序列化为字符串 - 也就是说,它不知道每个成员的类型,因此将所有内容包含在“”中。

有关此问题的详情,请参阅Creating JSON arrays in Boost using Property Trees

答案 5 :(得分:1)

所有需要明确为字符串定制翻译器的解决方案,对我来说似乎很容易出错,因为有时可能会忘记它。对于属性树的put方法,通过继承有某种重载方式来隐式处理该问题将是一件好事,但是以健壮的方式实现这是不可能的,因为它是模板,并且您必须确保树中所有方法的完全协方差。如果可能的话,通常也应避免将boost库的内容更改为解决方法。

到目前为止,我发现没有黑客的最可靠的方法是(自C ++ 11起):

  • 使用具有>
  • 的增强属性树
  • 为您的变体提供翻译器(详细信息,请参见下面的链接!),但请不要根据JSON特有的内容“黑客化”!这些几乎是完全正交的方面!
  • 为JSON编写自己的读取器和写入器,应该容易地与Boost的读取器和写入器相适应

优点:

  • 就JSON特定的详细信息而言,无需黑客即可影响属性树
  • 除了您自己的新类型(变体翻译器)的专业化以外,Boost库及其名称空间不会受到污染
  • 类型比基于自定义字符串的属性树方法更安全
  • 对于具有树的非频繁序列化的许多运行时场景,应该更快

缺点:

  • 需要一些努力才能获得非常小的行为细节
  • 在编译方面可能会慢一些
  • 对于频繁序列化和树的较小更改的运行时场景,可能会变慢(可以肯定地进行优化)
  • 将json读回树是某种有疑问的哲学工作,以确保所使用的类型之间尽可能地对称(出于许多目的,而不是学术问题)

有关更多详细信息,例如,参见

http://marko-editor.com/articles/property_tree_store_anything/

您可以轻松地对此进行修改以适应不同的用途。

答案 6 :(得分:0)

我最终在我的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();
    }
} }

希望有帮助。