Мне нужно получить вложенный объект внутри строки JSON, и я пытаюсь сделать это с помощью rapidjson. Все, что я нашел, — это как получить массивы и базовые типы, но не подобъекты. Я создал следующий игрушечный пример, который дает ошибку:
rapidjson::Document document;
std::string test = " { \"a\": { \"z\" : 21 } } ";
std::cout << test << std::endl;
if ( document.Parse<0>( test.c_str() ).HasParseError() ) {
std::cout << "Parsing error" << std::endl;
} else {
if ( document[ "a" ].IsObject() ) {
std::cout << "OK" << std::endl;
std::cout << document[ "a" ].GetString() << std::endl;
}
}
Это вывод при выполнении:
{ "a": { "z" : 21 } }
OK
JSONTest: ../rapidjson/document.h:441: const typename Encoding::Ch* rapidjson::GenericValue<Encoding, Allocator>::GetString() const [with Encoding = rapidjson::UTF8<char>, Allocator = rapidjson::MemoryPoolAllocator<rapidjson::CrtAllocator>]: Assertion `IsString()' failed. Aborted
Как мне извлечь внутренний объект, чтобы продолжить мой анализ? Благодарю.
редактироватьМне нужно получить строковое представление внутреннего объекта, чтобы я мог вызвать другую функцию, которая будет его анализировать.
Редактировать 2: код, который позволяет получить внутренний объект в виде строки:
rapidjson::Document document;
std::string test = "{\"a\":{\"z\":21}} ";
if ( document.Parse<0>( test.c_str() ).HasParseError() ) {
std::cout << "Error parsing" << std::endl;
} else {
if ( document[ "a" ].IsObject() ) {
rapidjson::StringBuffer sb;
rapidjson::Writer<rapidjson::StringBuffer> writer( sb );
document[ "a" ].Accept( writer );
std::cout << sb.GetString() << std::endl;
}
}
Вам нужно перебирать членов объекта вручную, так как GetString () работает только со строковыми членами, а document [«a»] является объектом. Вам нужно перебирать членов этого объекта, используя переменную MemberIterator. У меня не было практики в C * более 15 лет, поэтому я могу только дать общее представление о том, как это должно работать:
for (MemberIterator m = document["a"].MemberBegin(); m != document["a"].MemberEnd(); ++m) {
std::cout << m.name << " " << (m.IsNumber()?m.GetNumber():m.GetString()) << endl;
}
Также вы можете захотеть взглянуть на метод Accept (), который, похоже, возвращает строку JSON объекта, который вы ему дали.
Если элемент является объектом, вы можете просто получить доступ к его свойствам с помощью []:
for (SizeType i = 0; i < layers.Size(); i++){
cout << layers[i]["name"].GetString() << endl;
}
Есть еще один замечательный подход, реализованный в rapidjson — JSON Pointers. Они имеют экспериментальный статус и согласно документации должны быть включены в v.1.1. В любом случае этот подход выглядит как XPATH для XML, поэтому для получения вложенного значения мы можем использовать такой синтаксис, как
Value* tmpValue = GetValueByPointer(doc, "/result/job/blob");
Я попробовал эту функциональность, и, на мой взгляд, это лучше, чем итераторы.
Вы можете использовать указатель для получения подобъекта:
Value& a = *GetValueByPointer(document, "/a");
int z = a["z"].GetInt();
Вот один пример кода, чтобы получить вложенный объект как rapidjson::Document
объект.
Document get_nested(Document &d, std::string key){
rapidjson::StringBuffer buffer;
const char *key_ctr = key.c_str();
assert(d[key_ctr].IsObject());
rapidjson::Writer<rapidjson::StringBuffer> writer(buffer);
d[key_ctr].Accept(writer);
rapidjson::Document result;
rapidjson::StringStream s(buffer.GetString());
result.ParseStream(s);
return result;
}
Вы также можете использовать указатель Document:
Document *document= new Document();
document->parse( test.c_str());
и положить указатель значения и использовать его
Value *val= document;
val = &(*val)["a"];
val = &(*val)["z"];
cout << val->GetString();
Это то, над чем я недавно работал:
void enter(const Value &obj, size_t indent = 0) { //print JSON tree
if (obj.IsObject()) { //check if object
for (Value::ConstMemberIterator itr = obj.MemberBegin(); itr != obj.MemberEnd(); ++itr) { //iterate through object
const Value& objName = obj[itr->name.GetString()]; //make object value
for (size_t i = 0; i != indent; ++i) //indent
cout << " ";
cout << itr->name.GetString() << ": "; //key name
if (itr->value.IsNumber()) //if integer
std::cout << itr->value.GetInt() ;
else if (itr->value.IsString()) //if string
std::cout << itr->value.GetString();else if (itr->value.IsBool()) //if bool
std::cout << itr->value.GetBool();
else if (itr->value.IsArray()){ //if array
for (SizeType i = 0; i < itr->value.Size(); i++) {
if (itr->value[i].IsNumber()) //if array value integer
std::cout << itr->value[i].GetInt() ;
else if (itr->value[i].IsString()) //if array value string
std::cout << itr->value[i].GetString() ;
else if (itr->value[i].IsBool()) //if array value bool
std::cout << itr->value[i].GetBool() ;
else if (itr->value[i].IsObject()){ //if array value object
cout << "\n ";
const Value& m = itr->value[i];
for (auto& v : m.GetObject()) { //iterate through array object
if (m[v.name.GetString()].IsString()) //if array object value is string
cout << v.name.GetString() << ": " << m[v.name.GetString()].GetString();
else //if array object value is integer
cout << v.name.GetString() << ": " << m[v.name.GetString()].GetInt();
cout << "\t"; //indent
}
}
cout << "\t"; //indent
}
}
cout << endl;
enter(objName, indent + 1); //if couldn't find in object, enter object and repeat process recursively
}
}
}
Это может обработать любой тип дерева JSON. Все, что вам нужно сделать, это передать значение как таковое:
Value v = document.GetObject();
Value& m= v;
enter(m);
И вы сделали!