web-dev-qa-db-ja.com

JsonCppでオブジェクトを反復処理する

jsoncpp を使用してJSON文字列をデコードするC++アプリケーションがあります。次の関数を作成しましたが、トップレベルのオブジェクトのみが表示されます...

オブジェクトリスト全体をダンプするにはどうすればよいですか?

-関数-

SaveJSON( json_data ); 

bool CDriverConfigurator::PrintJSONTree( Json::Value & root, unsigned short depth /* = 0 */) 
{
    printf( " {type=[%d], size=%d} ", root.type(), root.size() ); 

    if( root.size() > 0 ) {
        for( Json::ValueIterator itr = root.begin() ; itr != root.end() ; itr++ ) {
            PrintJSONTree( itr.key(), depth+1 ); 
        }
        return true;
    }

    // Print depth. 
    for( int tab = 0 ; tab < depth; tab++) {
        printf( "-"); 
    }

    if( root.isString() ) {
        printf( " %s", root.asString().c_str() ); 
    } else if( root.isBool() ) {
        printf( " %d", root.asBool() ); 
    } else if( root.isInt() ) {
        printf( " %d", root.asInt() ); 
    } else if( root.isUInt() ) {
        printf( " %d", root.asUInt() ); 
    } else if( root.isDouble() ) {
        printf( " %f", root.asDouble() ); 
    }
    else 
    {
        printf( " unknown type=[%d]", root.type() ); 
    }


    printf( "\n" ); 
    return true;
}

---入力----

{
   "modules":[
      {
         "config":{
            "position":[
               129,
               235
            ]
         },
         "name":"Modbus Task",
         "value":{
            "DeviceID":"This is the name",
            "Function":"01_READ_COIL_STATUS",
            "Length":"99",
            "Scan":"111",
            "Type":"Serve"
         }
      },
      {
         "config":{
            "position":[
               13,
               17
            ]
         },
         "name":"Modbus Connection",
         "value":{
            "Baud":"9600",
            "timeout":"2.5"
         }
      },
      {
         "config":{
            "position":[
               47,
               145
            ]
         },
         "name":"Modbus Device",
         "value":{
            "DeviceID":"55"
         }
      },
      {
         "config":{
            "position":[
               363,
               512
            ]
         },
         "name":"Function Something",
         "value":{

         }
      },
      {
         "config":{
            "position":[
               404,
               701
            ]
         },
         "name":"Function Something",
         "value":{

         }
      }
   ],
   "properties":{
      "Blarrg":"",
      "description":"",
      "name":"Modbus"
   },
   "wires":[
      {
         "src":{
            "moduleId":1,
            "terminal":"modbus.connection.output"
         },
         "tgt":{
            "moduleId":2,
            "terminal":"modbus.connection.input"
         }
      },
      {
         "src":{
            "moduleId":2,
            "terminal":"modbus.device.output"
         },
         "tgt":{
            "moduleId":0,
            "terminal":"modbus.device.output"
         }
      },
      {
         "src":{
            "moduleId":3,
            "terminal":"dataOut"
         },
         "tgt":{
            "moduleId":4,
            "terminal":"dataIn"
         }
      },
      {
         "src":{
            "moduleId":3,
            "terminal":"dataIn"
         },
         "tgt":{
            "moduleId":0,
            "terminal":"data1"
         }
      }
   ]
}

-出力-

{type=[7], size=3} {type=[4], size=0} - modules
{type=[4], size=0} - properties
{type=[4], size=0} - wires 
25

JSONの再帰またはキー→値の性質、およびそれが使用しているライブラリにどのように関連するかについて、見かけ上はうまく処理されていないことに関連するいくつかのエラーがあります。このコードはまったくテストしていませんが、うまく機能するはずです。

void CDriverConfigurator::PrintJSONValue( const Json::Value &val )
{
    if( val.isString() ) {
        printf( "string(%s)", val.asString().c_str() ); 
    } else if( val.isBool() ) {
        printf( "bool(%d)", val.asBool() ); 
    } else if( val.isInt() ) {
        printf( "int(%d)", val.asInt() ); 
    } else if( val.isUInt() ) {
        printf( "uint(%u)", val.asUInt() ); 
    } else if( val.isDouble() ) {
        printf( "double(%f)", val.asDouble() ); 
    }
    else 
    {
        printf( "unknown type=[%d]", val.type() ); 
    }
}

bool CDriverConfigurator::PrintJSONTree( const Json::Value &root, unsigned short depth /* = 0 */) 
{
    depth += 1;
    printf( " {type=[%d], size=%d}", root.type(), root.size() ); 

    if( root.size() > 0 ) {
        printf("\n");
        for( Json::Value::const_iterator itr = root.begin() ; itr != root.end() ; itr++ ) {
            // Print depth. 
            for( int tab = 0 ; tab < depth; tab++) {
               printf("-"); 
            }
            printf(" subvalue(");
            PrintJSONValue(itr.key());
            printf(") -");
            PrintJSONTree( *itr, depth); 
        }
        return true;
    } else {
        printf(" ");
        PrintJSONValue(root);
        printf( "\n" ); 
    }
    return true;
}
28
Omnifarious

Json :: Valueを出力するだけの場合、そのための method があります。

Json::Value val;
/*...build the value...*/
cout << val.toStyledString() << endl;

また、Json::StyledWriter、そのドキュメントは here です。私はそれが人間に優しいバージョンを印刷すると信じています。また、Json::FastWriter、ドキュメント ここ 、よりコンパクトなフォームを印刷します。

7
JaredC

これはjsonオブジェクトとオブジェクトメンバー(およびその値)のいずれかを出力できる良い例です:

Json::Value root;               // Json root
Json::Reader parser;            // Json parser

// Json content
string strCarPrices ="{ \"Car Prices\": [{\"Aventador\":\"$393,695\", \"BMW\":\"$40,250\",\"Porsche\":\"$59,000\",\"Koenigsegg Agera\":\"$2.1 Million\"}]}";

// Parse the json
bool bIsParsed = parser.parse( strCarPrices, root );
if (bIsParsed == true)
{
    // Get the values
    const Json::Value values = root["Car Prices"];

    // Print the objects
    for ( int i = 0; i < values.size(); i++ )
    {
        // Print the values
        cout << values[i] << endl;

        // Print the member names and values individually of an object
        for(int j = 0; j < values[i].getMemberNames().size(); j++)
        {
            // Member name and value
            cout << values[i].getMemberNames()[j] << ": " << values[i][values[i].getMemberNames()[j]].asString() << endl;
        }
    }
}
else
{
    cout << "Cannot parse the json content!" << endl;
}

出力 :

{
        "Aventador" : "$393,695",
        "BMW" : "$40,250",
        "Koenigsegg Agera" : "$2.1 Million",
        "Porsche" : "$59,000"
}
Aventador: $393,695
BMW: $40,250
Koenigsegg Agera: $2.1 Million
Porsche: $59,000
1
BattleTested

Json :: valueのすべてのフィールドを反復処理する簡単な方法があります。 printfのものは省略しました。

#include "cpprest/json.h"
#include "cpprest/filestream.h"

using web::json::value;
using std::wstring;

static void printOneValue(const wstring &key, const double &value);
static void printOneValue(const wstring &key, const bool &value);
static void printOneValue(const wstring &key, const int &value);
static void printOneValue(const wstring &key, const wstring &value);
static void printOne(const wstring &key, const value &v, _num level);
static void printTree(const value &v);

static void printTree(const value &v)
{
    if(!v.is_object())
        return;

    try
    {
        printOne(wstring(), v, 0);
    }
    catch(...)
    {
        // error handling
    }
}

static void printOne(const wstring &key, const value &v, _num level)
{
    switch(v.type())
    {
    case value::value_type::Number:
        if(v.is_double())
            printOneValue(key, v.as_double());
        else
            printOneValue(key, v.as_integer());
        break;
    case value::value_type::Boolean:
        printOneValue(key, v.as_bool());
        break;
    case value::value_type::String:
        printOneValue(key, v.as_string());
        break;
    case value::value_type::Object:
        for(auto iter : v.as_object())
        {
            const wstring &k = iter.first;
            const value &val = iter.second;
            printOne(k, val, level+1);
        }
        break;
    case value::value_type::Array:
        for(auto it : v.as_array())
        {
            printOne(key, it, level+1);
        }
        break;
    case value::value_type::Null:
    default:
        break;
    }
}

static void printOneValue(const wstring &key, const wstring &value)
{
    // process your key and value
}

static void printOneValue(const wstring &key, const int &value)
{
    // process your key and value
}

static void printOneValue(const wstring &key, const double &value)
{
    // process your key and value
}

static void printOneValue(const wstring &key, const bool &value)
{
    // process your key and value
}
0
Sergey Merkelov