JSON

IT网络文摘的软件学习笔记

字号+ 作者:H5之家 来源:H5之家 2017-05-09 10:01 我要评论( )

【源码分析】cJSON库学习

【源码分析】cJSON库学习 发布时间:2017-05-05 17:03:27.173作者:IT网络文摘

cJSON库是什么?

cJSON是一个轻量级的json解析库。使用起来非常简单,整个库非常地简洁,核心功能的实现都在cJSON.c文件,非常适合阅读源代码来学习C语言。最近读完这个库的源码,分享自己收获的一些心得。

什么是json,照搬json官网的说法:

JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式。 易于人阅读和编写。同时也易于机器解析和生成。 它基于JavaScript Programming Language, Standard ECMA-262 3rd Edition - December 1999的一个子集。 JSON采用完全独立于语言的文本格式,但是也使用了类似于C语言家族的习惯(包括C, C++, C#, Java, JavaScript, Perl, Python等)。 这些特性使JSON成为理想的数据交换语言。

cJSON库里面有什么?

cjson库github地址:https://github.com/DaveGamble/cJSON
整个库包含cJSON.h和cJSON.c两个文件,头文件定义了一系列的API。这个库最基本也最重要的功能就是解析一个json字符串,使用的API是cJSON_Parse。cJSON_Parse函数调用了cJSON_ParseWithOpts函数,该函数实现了具体的逻辑。

两个函数的原型如下:

CJSON_PUBLIC(cJSON *) cJSON_Parse(const char *value); CJSON_PUBLIC(cJSON *) cJSON_ParseWithOpts(const char *value, const char **return_parse_end, cJSON_bool require_null_terminated);

函数接收一段字符串,然后进行解析后返回。解析完返回的是一个cjson结构,cJSON结构的定义如下:

typedef struct cJSON {    struct cJSON *next; // 向后指针     struct cJSON *prev; // 向前指针     struct cJSON *child; // 指向子元素,比如子数组或者子对象     int type; // 元素的类型     char *valuestring; // 元素的字符串值,如果type == cJSON_String 或者 type == cJSO_Raw     int valueint; // 已废弃,现在使用cJSON_SetNumberValue设置整型值     double valuedouble; // 元素的整型值,如果type == cJSON_Number     char *string; // 表示元素键值的值,如果它有子元素的话} cJSON;如何解析一个json字符串?

json的官网在这里,
网站首页描述了json是什么以及它的格式规范,有了规范之后,可以知道json是如何构成的,因此就有了如何解析json数据的方向。

json使用两种结构构建,对象或者数组。

对象使用{作开头,}作结尾,里边的每一个元素都是键值对的无序集合,键名和值使用:分隔,使用,分隔每一个元素;数组使用[作开头,]作结尾,里面的元素都是有序的值组成的集合,且使用,做分隔符。

每一个值可以是字符串,整型,也可以是true,false,null等常量,还可以是对象或数组,因为json结构是可嵌套的。

因此,我们可以得知:

1、可以根据json的首字母判断整个json的类型,如果json以'{'开头时,就是对象,以'['开头时,就是数组,否则就是字符串或者其他常量。

2、如果是对象,那么它的一定有键名,先解析它的键名,然后解析它的值,解析值的过程与第一步一样,递归解析

3、如果是数组,则逐个解析数组内的元素,直到遇到]为止,解析数组里面的元素的过程也是与第一步一致,递归解析。

这就是根据json官网的定义得出解析json字符串的思路,接下来看看cJSON库是如何实现的。cJSON_Parse的实现流程图如下:

Android培训,安卓培训,手机开发培训,移动开发培训,云培训培训

cJSON_ParseWithOpts函数里面调用了parse_value,是整个函数的核心实现。
parse_value函数的流程图如下所示:

Android培训,安卓培训,手机开发培训,移动开发培训,云培训培训

可以看到,parse_value是对json值的开头进行判断,然后进入相应的分支进行解析,下面对每一个分支进行分析。解析出来的值是保存在cJSON的结构体中,以下命名为item。

常量

如果json值是以'null','true','false',则分别将item的type设置为cJSON_NULL、cJSON_TRUE、cJSON_FALSE。然后继续解析剩下的json值。

string

如果遇到"开头,则说明json值是字符串,就解析它的值,此时只需要拿到两个"之间的值即可。保存字符串也是一个结构体,需要申请内存,计算长度的过程中,当遇到转义字符时,需要记录,因为转义符不保存。

number

当遇到数字开头时,将其后面的数字字符记录起来,然后转成整型数字,然后做值的范围检查。

array

解析数组时,为数组的元素创建一个新的json结构体new_item,然后继续解析数组里面的值,用','判断下一个元素的位置,得到的值保存到结构体中,并将多个元素用链表连接起来。一直解析,直到遇到']'符号。

object

解析对象的过程与数组的类似,为对象的元素创建一个新的json结构体new_item,然后继续解析对象里面的值,对象是有键值对组成的,因此先得到键的值,然后用':'判断值的位置,进而继续解析得到值,多个键值对之间用','分隔开,最后用链表连接起来。一直解析,直到遇到'}'符号。

其他

在解析所有值之前,会调用skip_whitespace函数过滤字符串两边的所有空白字符。此处是ASCII码小于等于32的字符,如:\t、\n。函数如下:

static const unsigned char *skip_whitespace(const unsigned char *in) {    while (in && *in && (*in <= 32))     {        in++;     }    return in; }总结

 

1.本站遵循行业规范,任何转载的稿件都会明确标注作者和来源;2.本站的原创文章,请转载时务必注明文章作者和来源,不尊重原创的行为我们将追究责任;3.作者投稿可能会经我们编辑修改或补充。

相关文章
  • blog:ssslinppp Spring学习笔记

    blog:ssslinppp Spring学习笔记

    2017-05-07 17:02

  • JSON的学习和使用

    JSON的学习和使用

    2017-05-07 17:02

  • Seajs的学习笔记

    Seajs的学习笔记

    2017-05-05 18:02

  • 人工智能和机器学习领域有哪些有趣的开源项目?

    人工智能和机器学习领域有哪些有趣的开源项目?

    2017-05-05 10:11

网友点评