Search code examples
c++jsonboostboost-propertytree

Using boost property tree to read int array


I have some JSON with a handful of integer array variables, like so:

{"a": [8, 6, 2], "b": [2, 2, 1]}

I would like to use boost property_tree, for instance:

std::stringstream ss;
boost::property_tree::ptree pt;

ss << "{\"a\": [8, 6, 2], \"b\": [2, 2, 1]}";

boost::property_tree::read_json(ss, pt);
std::vector<int> a = pt.get<std::vector<int> >("a");

This doesn't work, nor does any variation on an int pointer that I've tried. How may I read an array from a property tree?


Solution

  • JSON support, is spotty with boost property tree.

    The property tree dataset is not typed, and does not support arrays as such. Thus, the following JSON / property tree mapping is used:

    • JSON objects are mapped to nodes. Each property is a child node.
    • JSON arrays are mapped to nodes. Each element is a child node with an empty name. If a node has both named and unnamed child nodes, it cannot be mapped to a JSON representation.
    • JSON values are mapped to nodes containing the value. However, all type information is lost; numbers, as well as the literals "null", "true" and "false" are simply mapped to their string form.
    • Property tree nodes containing both child nodes and data cannot be mapped.

    (from the documentation)

    You can iterate the array with a helper function.

    #include <boost/property_tree/ptree.hpp>
    #include <boost/property_tree/json_parser.hpp>
    
    using boost::property_tree::ptree;
    
    template <typename T>
    std::vector<T> as_vector(ptree const& pt, ptree::key_type const& key)
    {
        std::vector<T> r;
        for (auto& item : pt.get_child(key))
            r.push_back(item.second.get_value<T>());
        return r;
    }
    
    int main()
    {
        std::stringstream ss("{\"a\": [8, 6, 2], \"b\": [2, 2, 1]}");
    
        ptree pt;
        read_json(ss, pt);
    
        for (auto i : as_vector<int>(pt, "a")) std::cout << i << ' ';
        std::cout << '\n';
        for (auto i : as_vector<int>(pt, "b")) std::cout << i << ' ';
    }
    

    See it Live On Coliru. Output:

    8 6 2 
    2 2 1