`
dowhathowtodo
  • 浏览: 781244 次
文章分类
社区版块
存档分类
最新评论

JSON格式解析和libjson使用简介

 
阅读更多

文章转载自http://dev.uphoneapp.com/doc/view.xhtml?id=52261
JSON格式解析和libjson使用简介


在阅读本文之前,请先阅读下《Rss Reader实例开发之系统设计》一文。

Rss Reader实例开发中,进行网络数据交换时主要使用到了两种数据格式:JSON与XML。本文主要介绍JSON格式的简单概念及JSON在Rss Reader中的应用,XML格式的使用将在下一篇文章做介绍。

JSON简介:


JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式,可以把JSON的结构理解成无序的、可嵌套的key-value键值对集合,这些key-value键值对是以结构体或数组的形式来组织的。同一级的key-value键值对之间是用一个“,”(逗号)隔开,每个key-value键值对是由一个key后面紧接一个“:”(冒号),冒号后面是这个key对应的value。Key是一个word,由大小写字母、下划线及数字组成,可以由双引号封闭,也可以不加双引号;而value的取值集为:Number、Boolean(true或false)、null、String、Object及Array,如图一:



(图一)

1、Number:数值,包括整形数与浮点数,如:123、0.83、-2.7e10。其结构如图二:


(图二)


2、String:字符串,是以双引号封闭起来的一串字符,使用反斜杠来转义,如:\\、\n等,JSON中字符串的概念与C/C++或者JAVA语言里的字符串概念差不多,如:”abc”。其结构如图三:


(图三)

3、Object:对象,也可理解成一个结构体,是以一对大括号封闭起来的无序的key-value键值对集合,例如:{name:"Susan", age:27, birthday:{year:1984, month:2, day:11}};也可以写成:{"name":"Susan", "age":27, "birthday":{"year":1984, "month":2, "day":11}};其结构如图四:


(图四)

4、Array:数组,JSON的数组是一个以中括号封闭起来的value的集合,即数组内的各个成员的数据类型可以不一样,这一点就跟C/JAVA的数组概念不同了。每个value之间是由一个“,”(逗号)隔开,例如:[123, abc, false, {name:mj}];其结构如图五:


(图五)


关于JSON的详细说明与教程请自行到网络上搜索,有很多。

下面我们就来动手写一个例子:

  1. {
  2. result:true,
  3. root:{
  4. version:"201007091640",
  5. channels:[
  6. {
  7. name:"新闻中心",
  8. subchnls:[
  9. {
  10. title:"焦点新闻",
  11. link:"http://news.mtc.sohu.com/news/channel/1/news.rss",
  12. desc:"家事、国事、天下事"
  13. },
  14. {
  15. title:"新闻频道",
  16. link:"http://news.mtc.sohu.com/news/channel/2/news.rss",
  17. desc:"让您实时掌握国际动态"
  18. },
  19. {
  20. title:"军事频道",
  21. link:"http://news.mtc.sohu.com/news/channel/3/news.rss",
  22. desc:"军事"
  23. }
  24. ]
  25. },
  26. {
  27. name:"体育新闻",
  28. subchnls:[
  29. {
  30. title:"体育要闻汇总",
  31. link:"http://news.mtc.sohu.com/news/channel/4/news.rss",
  32. desc:"erewr"
  33. },
  34. {
  35. title:"国际足坛",
  36. link:"http://news.mtc.sohu.com/news/channel/5/news.rss",
  37. desc:"werewr"
  38. }
  39. ]
  40. }
  41. ]
  42. }
  43. }


这段JSON描述了一个对象(最外层大括号包围的部分),为了方便区分,我们就将其称为对象A吧。对象A有两个Item(即key-value键值对),一个是result,其值为true;一个是root,其值为一个对象,称为对象B。对象B也有两个Item,一个是version,其值为一个字串” 201007091640”;一个是channels,其值是一个数组,而数组的成员都是一个对象,每个对象又包含两个Item,一个是name,值分别为字串"新闻中心"和"体育新闻";一个是subchnls,值都是数组,每个数组又分别有若干个成员,每个subchnls成员也都是一个对象,每个对象都有三个Item:title、link和desc。也许你看到这,已经是一头大汗了,不过没关系,我们来帖张这段JSON文本对应的结构图,有图就有真相,请看图六:


(图六:黑色实线为对象,虚线为值,橙色实线为数组)


在RssReader中使用cJSON:


在RssReader中使用了开源库cJSON来解析JSON,所以在此就介绍下cJSON的使用:

在CJSON中,一个key-value键值对被解析并存放在一个cJSON结构体变量中,其value取值集为:FALSE,TRUE,NULL,NUMBER,STRING,OBJECT,ARRAY。它们分别被存放在CJSON对象的child、valuestring、valueint、valuedouble变量中,而用于判断某个CJSON对象value的数据类型则是CJSON对象的type变量,其取值范围与CJSON对象的value集是一一对应的,如:cJSON_False对应FALSE。
cJSON Types:

  1. #definecJSON_False0
  2. #definecJSON_True1
  3. #definecJSON_NULL2
  4. #definecJSON_Number3
  5. #definecJSON_String4
  6. #definecJSON_Array5
  7. #definecJSON_Object6

cJSON 结构体:

  1. typedefstructcJSON
  2. {
  3. structcJSON*next,*prev;//指向上一项/下一项
  4. structcJSON*child;//指向下一级,也就是当type为cJSON_Object或cJSON_Array时,此指针不为空。
  5. inttype;
  6. char*valuestring;//当type为cJSON_String时
  7. intvalueint;//当type为cJSON_Number时
  8. doublevaluedouble;//当type为cJSON_Number时
  9. char*string;//当前项的名称,也就是key-value键值对的key
  10. }cJSON;

在解析JSON过程中,从JSON格式描述的value数据到CJSON对象中存放的变量的一个映射关系如图七:


(图七)


对CJSON格式的解析是使用cJSON_Parse()方法,其传入的参数是一个CJSON的Object/Array结构的字串,解析成功则返回一个cJSON结构体变量的指针,在使用完成后需要调用cJSON_Delete()将该指针销毁。CJSON是以树状结构来组织内部的各个cJSON结构体变量的,一般地,要使用某个cJSON结构体变量,需要调用cJSON_GetObjectItem()方法并根据其父节点的cJSON结构体变量指针与该项的名称来获取其指针,举个例子:

  1. boolbResult;
  2. charjsonString[]=“{result:true}”;
  3. //获取result的值true
  4. cJSON*pItem=NULL;
  5. cJSON*pRoot=cJSON_Parse(jsonString);
  6. if(pRoot)
  7. {
  8. pItem=cJSON_GetObjectItem(pRoot,“result”);
  9. if(pItem)
  10. {
  11. bResult=pItem->valueint;//由于result的值type为cJSON_False或cJSON_True,所以其值被存放在valueint变量中
  12. }
  13. cJSON_Delete(pRoot);
  14. }

在上例中,不管result的值type为何类型,都是通过调用cJSON_GetObjectItem()方法获取其对应的cJSON结构体变量的指针,只是在处理其对应的值时会有所不同。如果result的值type为cJSON_Object,则需要通过调用cJSON_GetObjectItem( pItem, “subItemKey”)来获取其子Item的指针。在处理值type为cJSON_Array的Item时,就需要再用到另外两个API:cJSON_GetArraySize ()和cJSON_GetArrayItem()。我们举个获取一个数组成员值的例子:

  1. char*out;
  2. charjsonString[]=“{colors:[\“red\”,\“green\”,\“blue\”,\“yellow\”,\“white\”]}”;
  3. cJSON*pArray=NULL;
  4. cJSON*pRoot=cJSON_Parse(jsonString);
  5. if(pRoot)
  6. {
  7. pArray=cJSON_GetObjectItem(pRoot,“colors”);
  8. if(pArray)
  9. {
  10. cJSON*pArrayItem=NULL;
  11. intnCount=cJSON_GetArraySize(pArray);//获取pArray数组的大小
  12. for(inti=0;i<nCount;i++)
  13. {
  14. pArrayItem=cJSON_GetArrayItem(pArray,i);
  15. out=cJSON_Print(pArrayItem);//将pArrayItem的值以字串的形式打印到char型buffer上,cJSON_Print()会自动分配内存空间,用完需要释放内存。
  16. SS_printf(“arrayitem%d:%s\n”,i,out);
  17. Free(out);
  18. }
  19. }
  20. cJSON_Delete(pRoot);
  21. }

在提取一个复杂的JSON格式的数据时,也只是将以上两个例子使用到的方法进行组合调用罢了。所以对CJSON提供的API的使用是很简单有效的。有了以上知识的了解,我们就可以编写一些代码将例一中的JSON解析并提取其中的数据,还是贴点代码才是硬道理,代码如下:

TChannelsData.h:

  1. /**子频道信息结构体
  2. *
  3. */
  4. structSUBCHNL_DATA
  5. {
  6. SUBCHNL_DATA();
  7. voidclear();
  8. TUChar*m_title;
  9. char*m_link;
  10. TUChar*m_desc;
  11. };
  12. /**大频道信息结构体
  13. *
  14. */
  15. structCHANNEL_DATA
  16. {
  17. CHANNEL_DATA();
  18. TUChar*m_pszTitle;
  19. vectorm_aSubChnlList;
  20. };
  21. //………….
  22. //TChannelsData类成员变量:RSSReaderConfig版本号
  23. charm_pszVersion[32];
  24. //TChannelsData类成员变量:频道信息列表
  25. vectorm_aChnlsList;
  26. //………….

TChannelsData.cpp:

  1. /**解析JSON格式的内容
  2. *
  3. *\parampszJsonText解析的JSON格式内容字串
  4. *
  5. *\returntrue:有更新数据;false:没有更新数据
  6. */
  7. BooleanTChannelsData::ParseJson(char*pszJsonText)
  8. {
  9. //char*out;
  10. cJSON*objJson;
  11. objJson=cJSON_Parse(pszJsonText);
  12. if(objJson)
  13. {
  14. //out=cJSON_Print(objJson);
  15. cJSON*objRootItem=NULL;
  16. //判断是否需要更新
  17. objRootItem=cJSON_GetObjectItem(objJson,"result");
  18. if(objRootItem)
  19. {
  20. if(!objRootItem->valueint)
  21. {
  22. returnFALSE;
  23. }
  24. }
  25. else
  26. {
  27. returnFALSE;
  28. }
  29. //获取更新数据,根节点root
  30. objRootItem=cJSON_GetObjectItem(objJson,"root");
  31. if(objRootItem)
  32. {
  33. cJSON*objJsonItem=NULL;
  34. //获取版本号
  35. objJsonItem=cJSON_GetObjectItem(objRootItem,"version");
  36. if(objJsonItem)
  37. {
  38. Int32nLen=strlen(objJsonItem->valuestring);
  39. strncpy(m_pszVersion,objJsonItem->valuestring,(nLen<32)?nLen:31);
  40. }
  41. //解析出大频道
  42. _ParseChannels(objRootItem);
  43. }
  44. //SS_printf("=======[parsejson]%s\n",out);
  45. cJSON_Delete(objJson);
  46. //free(out);
  47. }
  48. returnTRUE;
  49. }
  50. /**解析出大频道
  51. *
  52. *\parampCJsoncJSON对象指针
  53. *
  54. *\returnvoid
  55. */
  56. voidTChannelsData::_ParseChannels(cJSON*pCJson)
  57. {
  58. cJSON*pJsonArray=NULL;
  59. if(!pCJson)
  60. {
  61. return;
  62. }
  63. pJsonArray=cJSON_GetObjectItem(pCJson,"channels");
  64. if(pJsonArray)
  65. {
  66. cJSON*pArrayItem=NULL;
  67. cJSON*pJsonTemp=NULL;
  68. Int32nSize=cJSON_GetArraySize(pJsonArray);
  69. for(Int32i=0;i<nSize;i++)
  70. {
  71. pArrayItem=cJSON_GetArrayItem(pJsonArray,i);
  72. if(pArrayItem)
  73. {
  74. CHANNEL_DATAtChannelData;
  75. Int32nLen=0;
  76. //获取大频道名称
  77. tChannelData.m_pszTitle=_JsonGetTUString(pArrayItem,"name");
  78. //解析出子频道
  79. _ParseSubChnls(tChannelData.m_aSubChnlList,pArrayItem);
  80. //将大频道信息对象压入列表中
  81. m_aChnlsList.push_back(tChannelData);
  82. }
  83. else
  84. {
  85. continue;
  86. }
  87. }
  88. }
  89. }
  90. /**解析子频道
  91. *
  92. *\paramaSubChnlList存放子频道数据的vector对象
  93. *\parampCJsoncJSON对象指针
  94. *
  95. *\returnvoid
  96. */
  97. voidTChannelsData::_ParseSubChnls(vector&aSubChnlList,cJSON*pCJson)
  98. {
  99. cJSON*pJsonArray=NULL;
  100. if(!pCJson)
  101. {
  102. return;
  103. }
  104. pJsonArray=cJSON_GetObjectItem(pCJson,"subchnls");
  105. if(pJsonArray)
  106. {
  107. cJSON*pArrayItem=NULL;
  108. //cJSON*pJsonTemp=NULL;
  109. Int32nSize=cJSON_GetArraySize(pJsonArray);
  110. for(Int32i=0;i<nSize;i++)
  111. {
  112. pArrayItem=cJSON_GetArrayItem(pJsonArray,i);
  113. if(pArrayItem)
  114. {
  115. SUBCHNL_DATAtSubChnlData;
  116. Int32nLen=0;
  117. //gettitle
  118. tSubChnlData.m_title=_JsonGetTUString(pArrayItem,"title");
  119. //getlink
  120. tSubChnlData.m_link=_JsonGetString(pArrayItem,"link");
  121. //getdesc
  122. tSubChnlData.m_desc=_JsonGetTUString(pArrayItem,"desc");
  123. aSubChnlList.push_back(tSubChnlData);
  124. }
  125. }
  126. }
  127. }
  128. /**获取指定的cJSON对象的指定属性值
  129. *
  130. *\parampJsonItemcJSON对象指针
  131. *\parampszKeycJSON对象属性
  132. *
  133. *\return返回JSON对象的值,以TUChar字串形式返回
  134. */
  135. TUChar*TChannelsData::_JsonGetTUString(cJSON*pJsonItem,char*pszKey)
  136. {
  137. TUChar*pszValue=NULL;
  138. Int32nLen;
  139. cJSON*pJsonTemp=NULL;
  140. pJsonTemp=cJSON_GetObjectItem(pJsonItem,pszKey);
  141. if(pJsonTemp)
  142. {
  143. nLen=strlen(pJsonTemp->valuestring)+1;
  144. pszValue=newTUChar[nLen];
  145. if(pszValue)
  146. {
  147. MemSet(pszValue,0,nLen*sizeof(TUChar));
  148. TUString::StrUtf8ToStrUnicode(pszValue,(constChar*)pJsonTemp->valuestring);
  149. }
  150. }
  151. returnpszValue;
  152. }
  153. /**获取指定的cJSON对象的指定属性值
  154. *
  155. *\parampJsonItemcJSON对象指针
  156. *\parampszKeycJSON对象属性
  157. *
  158. *\return返回JSON对象的值,以char字串形式返回
  159. */
  160. char*TChannelsData::_JsonGetString(cJSON*pJsonItem,char*pszKey)
  161. {
  162. char*pszValue=NULL;
  163. Int32nLen;
  164. cJSON*pJsonTemp=NULL;
  165. pJsonTemp=cJSON_GetObjectItem(pJsonItem,pszKey);
  166. if(pJsonTemp)
  167. {
  168. nLen=strlen(pJsonTemp->valuestring)+1;
  169. pszValue=newchar[nLen];
  170. if(pszValue)
  171. {
  172. MemSet(pszValue,0,nLen);
  173. strncpy(pszValue,pJsonTemp->valuestring,nLen-1);
  174. }
  175. }
  176. returnpszValue;
  177. }
  178. /**获取指定的cJSON对象的指定属性值
  179. *
  180. *\parampJsonItemcJSON对象指针
  181. *\parampszKeycJSON对象属性
  182. *
  183. *\return返回JSON对象的值,以int32形式返回
  184. */
  185. Int32TChannelsData::_JsonGetInt(cJSON*pJsonItem,char*pszKey)
  186. {
  187. Int32nValue=0;
  188. cJSON*pJsonTemp=NULL;
  189. pJsonTemp=cJSON_GetObjectItem(pJsonItem,pszKey);
  190. if(pJsonTemp)
  191. {
  192. nValue=pJsonTemp->valueint;
  193. }
  194. returnnValue;
  195. }
  196. /**获取指定的cJSON对象的指定属性值
  197. *
  198. *\parampJsonItemcJSON对象指针
  199. *\parampszKeycJSON对象属性
  200. *
  201. *\return返回JSON对象的值,以Boolean形式返回
  202. */
  203. BooleanTChannelsData::_JsonGetBoolean(cJSON*pJsonItem,char*pszKey)
  204. {
  205. BooleanbValue=FALSE;
  206. cJSON*pJsonTemp=NULL;
  207. pJsonTemp=cJSON_GetObjectItem(pJsonItem,pszKey);
  208. if(pJsonTemp)
  209. {
  210. if(pJsonTemp->valueint)
  211. {
  212. bValue=TRUE;
  213. }
  214. }
  215. returnbValue;
  216. }


总结:


JSON的结构简约,所以使得JSON的文档的数据量比较小,比较适合用于网络数据的交换,而且对JSON文档的解析和数据提取的方法也很简单,方便程序员的使用,当然也正是因为JSON的结构简约,使得JSON的可读性与可编辑性会稍差于XML,所以JSON比较适合在较少有人工阅读和编辑的情况下使用。

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics