最近一个 C/C++ 项目要用到 JSON, 所以选用了 json-c 库. C 语言不像 PHP/Python 等动态语言, 可以无缝地将 JSON 数据结构转为自身的数据结构, 所以操作起来会有些麻烦.
首先是数据结构. 在 json-c 里, 所有的 JSON 数据结构都是 json_object 类型, 然后这个 json_object 结构有一个 type 字段, 表明当前的对象是什么类型, 比如整数, 数组, 字符串等等. 没错, 大多数动态语言都是用类似的方法处理动态类型.
json-c 有自己的引用计数的内存管理机制, json_object_get() 函数用于手动地增加对象的引用计数, 相对地, json_object_put() 用于手动地减少引用计数. 要使用 json-c 必须非常明白每一个函数, 是否会增加或者减少对象的引用计数, 以避免内存泄露.
比较特殊的是, 将某个通过 xxx_new_xxx 方法创建的新对象加到另一个对象(对象容器)中(作为另一个对象的一个字段的值, 或者另一个数组的一个元素), 都不会增加新对象的引用计数. 所以, 虽然你 “new” 了一个对象, 但这时, 你却不能把它 “delete” 掉. 当你 “delete” 掉那个对象容器时, 它会自动的减少自己子节点的引用计数.
为了方便操作, 我写了一些辅助方法. 如, 通过点号(或者斜杠)分隔的路径获取属性:
json_object *json_util_get(json_object *obj, const char *path);
路径可以用数字来表示数组的下标. 比如 json_util_get(obj, “result.0.name”), 或者获取 obj 对象的 result 字段, 而 result 是一个数组, 读取该数组的下标为 0 的元素, 这个元素又是一个对象, 最终返回该对象的 name 属性. 因为有时候, 我们并不需要一级一级地关心中间的路径是否存在, 而只关心我们想要的最终结果是否存在. 这个方法非常有用.
另外, 就是数字和表示数字的字符串的问题. 在 PHP 等动态语言中, 表示数字的字符串很多情况下就是数字, 所以很难假设对方会严格地按要求且不会疏忽地把应该是 string 对象当作 int 对象传过来, 所以要在 C 语言里把这两者都当作是 int. 这又是另一个方法:
int json_util_get_int(json_object *obj, const char *path, int defval);
如果指定的字段不是整数而是字符串, 则是尝试把它转为整数返回来.
json-c-0.9库的json_object_object_get()引发崩溃问题
2011-10-29 17:31:20| 分类: LINUX|字号 订阅
在linux下编译开源的json-c-0.9库,在json对象不符合标准格式时,引发segmentation fault问题,程序退出。没有任何响应,这实在是让人无语。
经发现与json_tokener_parse() 失败有很大关系。
1. 错误的情况
这里先复习一下json提取子对象的一般过程,先parse,再get object,
json_object *newobj=NULL;
newobj = json_tokener_parse(mystr);
if(NULL==newobj)
//错误
{
printf("json parse err!");
break;
}
json_object *sub_obj=json_object_object_get(newobj, "name333");
//因为name333不存在,所以出错! segmetation fault了,程序在这里直接崩溃
错在哪里呢?原来是在NULL==newobj这里。libjson这个库对parse错误的判断方法是is_error()宏,
#define is_error(ptr) ((unsigned long)ptr > (unsigned long)-4000L)
不得不抱怨一下这奇怪的判断方法,但是它就是这么规定的。
2. 正确的情况:
再写一个正确的:
const char *mystr="{\"studentid\":\"123\", \"name\":\"Jacky\", \"info\":[{\"age\":18},{\"sex\":\"male\"}]}";
new_obj = json_tokener_parse(mystr);
if( is_error(new_obj) )
//正确
{
printf("oh, my god. json_tokener_parse err\n");
}
else
{
printf("object parse okay\n");
json_object *sub1obj = json_object_object_get(new_obj, "name");
if(
NULL==sub1obj) //这里就要用NULL判断了, 得不到对应的object就是NULL
{
printf("sub1obj err\n");
}
else
{
printf("sub1obj:%s\n", json_object_to_json_string(sub1obj));
json_object_put(sub1obj);
}
//----------------
json_object *sub2obj = json_object_object_get(new_obj, "info");
if(NULL==sub2obj)
{
printf("sub2obj err\n");
}
else
{
printf("sub2obj:%s\n", json_object_to_json_string(sub2obj));
json_object_put(new_obj);
}
}//else end