json-c-0.9库的json_object_object_get()引发崩溃问题

来源:互联网 发布:云盘 mac 编辑:程序博客网 时间:2024/05/22 04:25

最近一个 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

原创粉丝点击