STM32—cJson库解析与构造JSON数据

见资如面 2024-08-14 12:05:03 阅读 84

文章目录

一、 JSON简介二、 cJSON简介使用cJSON时的几个注意点可能会遇到的问题使用cJSON时的内存问题cJSON结构体cJSON常用库函数

三、使用cJSON构造JSON四、使用cJSON解析JSON

一、 JSON简介

JSON(JavaScriptObject Notation)是一种轻量级的数据交换格式。

它基于JavaScript的一个子集。JSON采用完全独立于语言的文本格式,但是也使用了类似于C语言家族的习惯。

这些特性使JSON成为理想的数据交换语言。易于人阅读和编写,同时也易于机器解析和生成,有效的提升了网络传输效率。

二、 cJSON简介

cJSON是一个超轻巧,方便,单文件,简单的可以作为ANSI-C标准的JSON解析器。由于c语言中,没有直接的字典,字符串数组等数据结构,所以要借助结构体定义,处理jsoncJSON对象的实现采用了树形结构,每个对象是树的一个节点,每个节点由cJSON这个结构体组成,对象中的元素也由cJSON这个结构体组成。同一层的对象和元素是双向链表结构,由next和prev指针链接。不同层的对象或元素由child指针链接起来。cJSON.h中有详细的注释cjson.h文件中包含了对于JSON格式的结构体定义以及一些操作JSON格式的功能函数,包括创建JSON、向JSON格式中添加数字,字符,布尔值等、读取JSON格式、将JSON格式转化为字符串等。cjson.c文件中就是功能函数的具体实现。下载连接:https://github.com/DaveGamble/cJSON

使用cJSON时的几个注意点

CJSON需要较多的堆栈空间,CJSON官方说:跑完他的test大概需要3k的空间。在单片机或者资源较少的地方使用CJSON时,注意空间分配 每一个CJOSN结构体都是一个比较大的空间,么使用完之后要及时delete,一旦子对象被添加到父对象之后,删除父对象就会删除子对象,所以一旦我们删除了父对象再删除子对象会出现问题。

可能会遇到的问题

对于绝大多数的STM32开发板,创建JSON对象、添加数据、将JSON转换为字符串是不会有问题的。

但是,使用cJSON_Parse()函数时会出错,什么数据都读不出来,究其原因,还是开发板内存不够,经不起JSON的折腾。

解决办法:修改启动文件中的Stack_Size和Heap_Size 。把堆栈的内存调大一些,再去编译基本就通过了。

使用cJSON时的内存问题

及时释放内存

构造JSON数据时,特别注意:cJSON_Print() 这种转换函数,会自动为指针申请空间,使用完之后一定要及时释放空间:free()cJSON的所有操作都是基于链表的,所以cJSON在使用过程中大量的使用malloc从堆中分配动态内存的,所以在使用完之后,应当及时调用下面的函数,清空cJSON指针所指向的内存,该函数也可用于删除某一条数据:

<code>(void) cJSON_Delete(cJSON *item);//该函数删除一条JSON数据时,如果有嵌套,会连带删除。

#include <stdio.h>

#include <stdlib.h>

#include "cJSON.h"

int main (int argc, const char * argv[])

{

// 创建JSON Object根数据项,之后便可向该根数据项中添加string或int等内容

cJSON *root = cJSON_CreateObject();

// 加入节点(键值对),节点名称为value,节点值为123.4

cJSON_AddNumberToObject(root,"value",123.4);

// 打印JSON数据包。cJSON_Print函数可以打印根数据项,加入制表符换行符等标识符使得JSON数据包更易阅读

char *out = cJSON_Print(root);

printf("%s\n",out);

// //释放json结构体

cJSON_Delete(root);

free(out);// 使用free函数释放被out字符串占用的内存空间

return 0;

}

内存钩子:cJSON在支持自定义malloc函数和free函数,方法如下:

// 使用cJSON_Hooks来连接自定义malloc函数和free函数:

typedef struct cJSON_Hooks

{

/* malloc/free are CDECL on Windows regardless of the default calling convention of the compiler, so ensure the hooks allow passing those functions directly. */

void *(CJSON_CDECL *malloc_fn)(size_t sz);

void (CJSON_CDECL *free_fn)(void *ptr);

} cJSON_Hooks;

(void) cJSON_InitHooks(cJSON_Hooks* hooks); // 初始化钩子cJSON_Hooks

cJSON结构体

//type一共有7种取值

#define cJSON_Invalid (0)

#define cJSON_False (1 << 0)

#define cJSON_True (1 << 1)

#define cJSON_NULL (1 << 2)

#define cJSON_Number (1 << 3)

#define cJSON_String (1 << 4)

#define cJSON_Array (1 << 5)

#define cJSON_Object (1 << 6)

#define cJSON_Raw (1 << 7) /* raw json */

//若是Number类型,则valueint或valuedouble中存储着值,若你期望的是int,则访问valueint,若期望的是double,则访问valuedouble,可以得到值。

#define cJSON_IsReference 256

#define cJSON_StringIsConst 512

typedef struct cJSON {

struct cJSON *next,*prev; /* 遍历数组或对象链的前向或后向链表指针*/

struct cJSON *child; //cJOSN结构体为一个双向列表,并可通过child指针访问下一层,对象的孩子节点

int type; //决定key的类型,数据项可以是字符串可以是整形,也可以是浮点型

char *valuestring; //若key的type是String,那么值就存在valuestring中,访问valuestring得到值

int valueint; //整数值

double valuedouble; //键值是浮点型的话,可以从valuedouble取出

char *string; //key名称

} cJSON;

cJSON常用库函数

//构造Json时会用到的函数

CJSON_PUBLIC(cJSON *) cJSON_CreateObject(void);//创建对象---常用

CJSON_PUBLIC(cJSON *) cJSON_CreateArray(void);//创建数组---常用

CJSON_PUBLIC(cJSON *) cJSON_CreateIntArray(const int *numbers, int count);//创建整型数组

CJSON_PUBLIC(cJSON *) cJSON_CreateDoubleArray(const double *numbers, int count);//创建双浮点型数组

CJSON_PUBLIC(cJSON*) cJSON_AddNullToObject(cJSON * const object, const char * const name);//在对象中添加null

CJSON_PUBLIC(cJSON*) cJSON_AddTrueToObject(cJSON * const object, const char * const name);//在对象中添加true

CJSON_PUBLIC(cJSON*) cJSON_AddFalseToObject(cJSON * const object, const char * const name);//在对象中添加false

CJSON_PUBLIC(cJSON*) cJSON_AddNumberToObject(cJSON * const object, const char * const name, const double number);//在对象中添加数字

CJSON_PUBLIC(cJSON*) cJSON_AddStringToObject(cJSON * const object, const char * const name, const char * const string);//在对象中添加字符串

CJSON_PUBLIC(cJSON_bool) cJSON_AddItemToObject(cJSON *object, const char *string, cJSON *item);//在对象中添加项目

CJSON_PUBLIC(cJSON_bool) cJSON_AddItemToArray(cJSON *array, cJSON *item);//在数组中添加项目

CJSON_PUBLIC(char *) cJSON_Print(const cJSON *item);//JSON数据结构转换为JSON字符串---有格式

CJSON_PUBLIC(char *) cJSON_PrintUnformatted(const cJSON *item);//JSON数据结构转换为JSON字符串---无格式

CJSON_PUBLIC(void) cJSON_Delete(cJSON *item); //清除结构体

//解析Json时会用到的函数

cJSON *cJSON_Parse(const char *value);

/*作用:将一个JSON数据包,按照cJSON结构体的结构序列化整个数据包,并在堆中开辟一块内存存储cJSON结构体

返回值:成功返回一个指向内存块中的cJSON的指针,失败返回NULL*/

cJSON *cJSON_GetObjectItem(cJSON *object,const char *string);

/*作用:获取JSON字符串字段值

返回值:成功返回一个指向cJSON类型的结构体指针,失败返回NULL*/

int cJSON_GetArraySize(cJSON *array);

/*作用:获取数组成员对象个数

返回值:数组成员对象个数*/

void cJSON_Delete(cJSON *c);

/*作用:释放位于堆中cJSON结构体内存

返回值:无*/

三、使用cJSON构造JSON

示例

#include <stdio.h>

#include "cJSON.h"

int main(void)

{

double grade[4]={ 66.51,118.52,61.53,128.54};

inttime[4]={ 123,456,789,150};

cJSON *TCP = cJSON_CreateObject();//创建一个对象

cJSON_AddStringToObject(TCP,"name","MQ");//添加字符串

cJSON_AddNumberToObject(TCP,"age",25); //添加整型数字

cJSON_AddNumberToObject(TCP,"height",183.52);//添加浮点型数字

cJSON_AddFalseToObject(TCP,"gender");//添加逻辑值false

cJSON *ADD= cJSON_CreateObject();//创建一个对象

cJSON_AddStringToObject(ADD,"country","China");//添加字符串

cJSON_AddNumberToObject(ADD,"zip-code",123456);//添加整型数字

cJSON_AddItemToObject(TCP,"address",ADD);

cJSON *SUB = cJSON_CreateArray();//创建一个数组

cJSON_AddStringToObject(SUB,"","政治"); //添加字符串到数组

cJSON_AddStringToObject(SUB,"","数学");

cJSON_AddStringToObject(SUB,"","英语");

cJSON_AddStringToObject(SUB,"","专业课");

cJSON_AddItemToObject(TCP,"subject",SUB);//添加数组到对象

cJSON *TIM = cJSON_CreateIntArray(time,4);//创建一个整型数组

cJSON_AddItemToObject(TCP,"time",TIM);

cJSON *GRA = cJSON_CreateDoubleArray(grade,4);//创建一个双浮点型数组

cJSON_AddItemToObject(TCP,"grade",GRA);

cJSON *STU = cJSON_CreateArray();//创建一个数组

cJSON *Z3 = cJSON_CreateObject();//创建一个对象

cJSON_AddStringToObject(Z3,"name","张三");//添加字符串

cJSON_AddNumberToObject(Z3,"age",24); //添加整型数字

cJSON_AddTrueToObject(Z3,"gender");//添加逻辑值

cJSON_AddItemToArray(STU,Z3);//添加对象到数组中

cJSON *L4 = cJSON_CreateObject();//创建一个对象

cJSON_AddStringToObject(L4,"name","李四");//添加字符串

cJSON_AddNumberToObject(L4,"age",25); //添加整型数字

cJSON_AddTrueToObject(L4,"gender");//添加逻辑值

cJSON_AddItemToArray(STU,L4);//添加对象到数组中

cJSON *W5 = cJSON_CreateObject();//创建一个对象

cJSON_AddStringToObject(W5,"name","王五");//添加字符串

cJSON_AddNumberToObject(W5,"age",26); //添加整型数字

cJSON_AddTrueToObject(W5,"gender");//添加逻辑值

cJSON_AddItemToArray(STU,W5);//添加对象到数组中

cJSON_AddItemToObject(TCP,"student",STU);//添加数组到对象中

char *json_data = cJSON_Print(TCP);//JSON数据结构转换为JSON字符串

printf("%s\n",json_data);//输出字符串

free(json_data);

cJSON_Delete(TCP);//清除结构体

return 0;

}

结果

{

"name": "MQ",//字符串

"age":25,//整数

"height": 183.5,//浮点数

"gender": false,//逻辑值

"address":{ "country": "China",

"zip-code": 123456

},//对象

"subject": ["政治", "数学", "英语", "专业课"],//字符型数组

"time": [123, 456, 789, 150], //整型数组

"grade": [66.51, 118.52, 61.53, 128.54],//浮点型数组

"student":[

{ "name":"张三","age":24,"gender":true},

{ "name":"李四","age":25,"gender":true},

{ "name":"王五","age":26,"gender":true}

]//对象型数组

}

四、使用cJSON解析JSON

实例:从中提取出我们想要获取的数据,然后进行分析和处理

#include <stdio.h>

#include "cJSON.h"

int main()

{

char json_string[]="{\"name\":\"MQ\",\"age\":25,\"height\":183.5,\"gender\":false,\

\"address\":{\"country\":\"China\",\"zip-code\":123456},\

\"subject\":[\"政治\",\"数学\",\"英语\",\"专业课\"],\

\"time\":[123,456,789,150],\"grade\":[66.51,118.52,61.53,128.54],\

\"student\":[{\"name\":\"张三\",\"age\":24,\"gender\":false},\

{\"name\":\"李四\",\"age\":25,\"gender\":true},\

{\"name\":\"王五\",\"age\":26,\"gender\":null}]}";//定义JSON字符串

cJSON* cjson = cJSON_Parse(json_string);//将JSON字符串转换成JSON结构体

char *json_data = cJSON_Print(cjson);//JSON数据结构转换为JSON字符串

if(cjson == NULL)//判断转换是否成功

{

printf("cjson error...\r\n");

}

else

{

printf("%s\n",json_data);//输出字符串

//打包成功调用cJSON_Print打印输出

}

printf("/*********************以下就是提取的数据**********************/\n");

char *name = cJSON_GetObjectItem(cjson,"name")->valuestring;//解析字符串

printf("%s\n",name);

int age = cJSON_GetObjectItem(cjson,"age")->valueint;//解析整型

printf("%d\n",age);

double height = cJSON_GetObjectItem(cjson,"height")->valuedouble;//解析双浮点型

printf("%.1f\n",height);

int gender = cJSON_GetObjectItem(cjson,"gender")->type; //解析逻辑值---输出逻辑值对应的宏定义数值

printf("%d\n",gender);

cJSON* ADD = cJSON_GetObjectItem(cjson,"address");//解析对象

char * country = cJSON_GetObjectItem(ADD,"country")->valuestring;//解析对象中的字符串

printf("%s\n",country);

int zip = cJSON_GetObjectItem(ADD,"zip-code")->valueint;//解析对象中的整型数字

printf("%d\n",zip);

cJSON* SUB = cJSON_GetObjectItem(cjson,"subject");//解析数组

int SUB_size = cJSON_GetArraySize(SUB);//获取数组成员个数

int i=0;

for(i=0;i<SUB_size;i++)

{

printf("%s ",cJSON_GetArrayItem(SUB,i)->valuestring);//解析数组中的字符串

}

printf("\n");

cJSON* TIM = cJSON_GetObjectItem(cjson,"time");//解析数组

int TIM_size = cJSON_GetArraySize(TIM);//获取数组成员个数

for(i=0;i<TIM_size;i++)

{

printf("%d ",cJSON_GetArrayItem(TIM,i)->valueint);//解析数组中的整型数字

}

printf("\n");

cJSON* GRA = cJSON_GetObjectItem(cjson,"grade");//解析数组

int GRA_size = cJSON_GetArraySize(GRA);//获取数组成员个数

for(i=0;i<GRA_size;i++)

{

printf("%f ",cJSON_GetArrayItem(GRA,i)->valuedouble);//解析数组中的浮点型数字

}

printf("\n");

cJSON* STU = cJSON_GetObjectItem(cjson,"student");//解析数组

int STU_size = cJSON_GetArraySize(STU);//获取数组成员个数

cJSON* STU_item = STU->child;//获取子对象

for(i=0;i<STU_size;i++)

{

printf("%s ",cJSON_GetObjectItem(STU_item,"name")->valuestring);//解析数组中对象中的字符串

printf("%d ",cJSON_GetObjectItem(STU_item,"age")->valueint);//解析数组中对象中的整型数字

printf("%d\n",cJSON_GetObjectItem(STU_item,"gender")->type);//解析数组中对象中的逻辑值---输出逻辑值对应的宏定义数值

STU_item = STU_item->next;//跳转到下一个对象中

}

free(json_data);

cJSON_Delete(cjson);//清除结构体

return 0;

}

结果

{

"name": "MQ",//字符串

"age":25,//整数

"height": 183.5,//浮点数

"gender": false,//逻辑值

"address":{ "country": "China",

"zip-code": 123456

},//对象

"subject": ["政治", "数学", "英语", "专业课"],//字符型数组

"time": [123, 456, 789, 150], //整型数组

"grade": [66.51, 118.52, 61.53, 128.54],//浮点型数组

"student":[

{ "name":"张三","age":24,"gender":false},

{ "name":"李四","age":25,"gender":true},

{ "name":"王五","age":26,"gender":null}

]//对象型数组

}

/*********************以下就是提取的数据**********************/

MQ

25

183.5

1

China

123456

政治 数学 英语 专业课

123 456 789 150

66.510000 118.520000 61.530000 128.540000

张三 24 1

李四 25 2

王五 26 4



声明

本文内容仅代表作者观点,或转载于其他网站,本站不以此文作为商业用途
如有涉及侵权,请联系本站进行删除
转载本站原创文章,请注明来源及作者。