转载地址:http://blog.csdn.net/dengjin20104042056/article/details/47831011
1.JSON格式简述
JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式。易于人阅读和编写,同时也易于机器解析和生成。它基于JavaScript(Standard ECMA-262 3rd Edition - December 1999)的一个子集。 JSON采用完全独立于语言的文本格式,但是也使用了类似于C语言家族的习惯(包括C, C++, C#, Java, JavaScript, Perl, Python等)。这些特性使JSON成为理想的数据交换语言。
cJSON是一个超轻巧,携带方便,单文件,简单的可以作为ANSI-C标准的JSON解析器。
接触yeelink平台之后,慢慢接触到JSON格式,虽然一些简单的情况可以通过string库函数解析和组装JSON数据包,但是若有cJSON库的帮助,解析和组装JSON数据包的工作便会变得简单的多,下面就从两个例子出发说明cJSON数据包的使用方法。
2.JSON结构体
熟悉使用cJSON库函数可从cJSON结构体入手,cJSON结构体如下所示:
cJSon类型
/* cJSON Types: */
#define cJSON_False 0
#define cJSON_True 1
#define cJSON_NULL 2
#define cJSON_Number 3
#define cJSON_String 4
#define cJSON_Array 5
#define cJSON_Object 6
#define cJSON_IsReference 256
#define cJSON_StringIsConst 512
#define cJSON_False 0
#define cJSON_True 1
#define cJSON_NULL 2
#define cJSON_Number 3
#define cJSON_String 4
#define cJSON_Array 5
#define cJSON_Object 6
#define cJSON_IsReference 256
#define cJSON_StringIsConst 512
说明
1.cJOSN结构体为一个双向列表,并可通过child指针访问下一层。
2.type变量决定数据项类型(键的类型),数据项可以是字符串可以是整形,也可以是浮点型。如果是整形值的话可从valueint,如果是浮点型的话可从valuedouble取出,以此类推。
3.string可理解为节点的名称,综合此处的第2点可理解为“键”的名称。
3.解析JSON数据包
例如在QCA平台中读取一个开关量的结果,向QCA平台请求之后可以获得以下JSON数据包:
{"timestamp":"2013-11-19T08:50:11","value":1}
在这个JSON数据包中有两个数据项(键值对),一个是时间戳,该时间戳为字符串形式;另一个是开关值,该开关值为整型。该例子主要用于模拟向QCA平台请求开关量数据。
参考代码
运行结果
value:1
2013-11-19T08:50:11
若干说明
1.cJSON_Parse函数负责解析JSON数据包,并按照cJSON结构体的结构序列化整个数据包。使用该函数会通过malloc函数在内存中开辟一个空间,使用完成需要手动释放。
2.cJSON_GetObjectItem函数可从cJSON结构体中查找某个子节点名称(键名称),如果查找成功可把该子节点序列化到cJSON结构体中。
3.如果需要使用cJSON结构体中的内容,可通过cJSON结构体中的valueint和valuestring取出有价值的内容(即键的值)
4.可通过cJSON_Delete释放内存空间。
4.组装JSON数据包
组装数据包的过程和解析数据包的过程相反,下面的例子描述如何组装以下数据包,该数据包只有一个数据项(键值对)。该例子主要用于模拟向QCA平台上传传感器数据。
{
"value": 123.400000
}
参考代码
若干说明
1. cJSON_CreateObject函数可创建一个根数据项,之后便可向该根数据项中添加string或int等内容
2. cJSON_AddNumberToObject向节点中添加子节点,例如此处添加value节点,节点值为123.4
3. cJSON_Print函数可以打印根数据项,加入制表符换行符等标识符使得JSON数据包更易阅读
4. 使用free函数释放被out占用的内存空间
1.重要函数说明
【1】两个创建
【创建JSON对象】cJSON *cJSON_CreateObject(void);
【创建JSON数组】cJSON *cJSON_CreateArray(void);
【2】两种添加
【向对象中添加】void
cJSON_AddItemToObject(cJSON *object,const char *string,cJSON *item);
【向数组中添加】void cJSON_AddItemToArray(cJSON *array, cJSON *item);
【3】常用几招
【向对象中增加数字】cJSON_AddItemToObject(root, "value", cJSON_CreateNumber(value));
【向对象中增加文件】cJSON_AddItemToObject(root, "string", cJSON_CreateString(string));
【4】JSON嵌套
【向对象中增加数组】cJSON_AddItemToObject(root, "rows", rows = cJSON_CreateArray());
【向数组中增加对象】cJSON_AddItemToArray(rows, row = cJSON_CreateObject());
2.创建各种各样的JSON数据包
在这里通过代码举几个例子,更多的内容请查看代码仓库中的相关文件。
【1】JSON数字
【简单说明】
【1】cJSON_AddItemToObject(root, "value", cJSON_CreateNumber(value));
【2】cJSON_AddNumberToObject(root, "value", value);
【1】和【2】效果完全相同。
【2】JSON字符串
【简单说明】
【1】 cJSON_AddItemToObject(root, "name", cJSON_CreateString(name));
【2】 cJSON_AddStringToObject(root, "name",name);
【1】和【2】效果完全相同。
【3】JSON布尔类型
【简单说明】
【1】布尔类型不需要加引号。
3.JSON格式嵌套
JSON格式在使用时往往存在嵌套,例如JSON对象中嵌套JSON数组而JSON数组中嵌套JSON对象,下面就通过几个简单的例子说明问题。
【1】JSON简单嵌套
【2】JSON复杂嵌套
【简单说明】
rows为JSON对象,rows对象中嵌套JSON数组,每一个JSON数组的元素又是一个JSON对象,该该对象名为row,row对象中具有两个键值对,分别是titile和url。
2015/08/19 17:50
【其他参考示例】
1. 函数的使用
宏的定义原型:
/* Macros for creating things quickly. */
#define cJSON_AddNullToObject(object,name) cJSON_AddItemToObject(object, name, cJSON_CreateNull())
#define cJSON_AddTrueToObject(object,name) cJSON_AddItemToObject(object, name, cJSON_CreateTrue())
#define cJSON_AddFalseToObject(object,name) cJSON_AddItemToObject(object, name, cJSON_CreateFalse())
#define cJSON_AddBoolToObject(object,name,b) cJSON_AddItemToObject(object, name, cJSON_CreateBool(b))
#define cJSON_AddNumberToObject(object,name,n) cJSON_AddItemToObject(object, name, cJSON_CreateNumber(n))
#define cJSON_AddStringToObject(object,name,s) cJSON_AddItemToObject(object, name, cJSON_CreateString(s))
示例:
cJSON *root = cJSON_CreateObject();
//先创建一个对象
cJSON_AddNumberToObject(root, "value", 123.4);
cJSON_AddStringToObject(root, "year", "2015");
cJSON_AddNullToObject(root, "secret");
cJSON_AddTrueToObject(root, "Bool");
cJSON_AddFalseToObject(root, "false");
cJSON_AddBoolToObject(root, "Yes", 3);
cJSON_AddBoolToObject(root, "Yes", 0);
out = cJSON_Print(root);
//打印对象
//out = cJSON_PrintUnformatted(root);
//非格式化的打印
//printf("===> 2: %s\n", out);
printf("%s\n", out);
cJSON_Delete(root);
//释放对象所占的内存
free(out);
函数的原型:
/* These calls create a cJSON item of the appropriate type. */
extern cJSON *cJSON_CreateNull(void);
extern cJSON *cJSON_CreateTrue(void);
extern cJSON *cJSON_CreateFalse(void);
extern cJSON *cJSON_CreateBool(int b);
extern cJSON *cJSON_CreateNumber(double num);
extern cJSON *cJSON_CreateString(const char *string);
extern cJSON *cJSON_CreateArray(void);
extern cJSON *cJSON_CreateObject(void);
示例:参考上面博客的代码
函数原型:
/* These utilities create an Array of count items. */
extern cJSON *cJSON_CreateIntArray(const int *numbers,int count);
extern cJSON *cJSON_CreateFloatArray(const float *numbers,int count);
extern cJSON *cJSON_CreateDoubleArray(const double *numbers,int count);
extern cJSON *cJSON_CreateStringArray(const char **strings,int count);
/* Append item to the specified array/object. */
extern void cJSON_AddItemToArray(cJSON *array, cJSON *item);
extern void cJSON_AddItemToObject(cJSON *object,const char *string,cJSON *item);
extern void cJSON_AddItemToObjectCS(cJSON *object,const char *string,cJSON *item);
示例:
char *out = NULL;
int array[] = {1, 2, 3, 4};
int array1[] = {1, 2, 3, 4, 5, 6};
cJSON *json = NULL;
cJSON *item = NULL;
cJSON *tmp = NULL;
json = cJSON_CreateObject();
//cJSON_AddItemToObject(json,"array", cJSON_CreateIntArray(array, 4));
//与下面两行等价
item = cJSON_CreateIntArray(array, 4);
cJSON_AddItemToObject(json, "array", item);
out = cJSON_Print(json);
printf("%s\n", out);
printf("=========================\n");
printf("size: %d\n", cJSON_GetArraySize(item));
//输出为4 4个元素
tmp = cJSON_GetArrayItem(json, 0);
//获取json中的第一个节点
if (NULL != tmp)
{
printf("ok..\n");
printf("size: %d\n", cJSON_GetArraySize(tmp));
}
else
printf("error...\n");
//此时item数组中的元素有 1 2 3 4 5 6 7
printf("=========================\n");
cJSON_AddItemToArray(item, cJSON_CreateNumber(5));
cJSON_AddItemToArray(item, cJSON_CreateNumber(6));
cJSON_AddItemToArray(item, cJSON_CreateNumber(7));
printf("size: %d\n", cJSON_GetArraySize(item));
out = cJSON_Print(json);
printf("%s\n", out);
tmp = cJSON_CreateIntArray(array1, 6);
//cJSON_AddItemToObject(json, "array1", tmp); //好像与下面一句话等价呢
cJSON_AddItemReferenceToObject(json, "array1", tmp);
out = cJSON_Print(json);
printf("%s\n", out);
//整形数组的参考示例
int array3[] = {1, 3, 5, 7, 9};
json = cJSON_CreateIntArray(array3, 5);
out = cJSON_Print(json);
cJSON_Delete(json);
printf("%s\n", out);
free(out);
//字符指针数组
const char *strings[7]={"Sunday","Monday","Tuesday","Wednesday","Thursday","Friday","Saturday"};
json = cJSON_CreateStringArray(strings, 7);
out = cJSON_Print(json);
cJSON_Delete(json);
printf("%s\n", out);
free(out);
5.总结
整体来说cJSON简单易用,解决了JSON数据包解析和组装的问题,几乎就掌握了CJSON。