目录
前言
一、JSON和cJson
二、cJSON是如何表示JSON数据的
三、如何封装完整的JSON数据
1. 先将串口打通,方便电脑查看log日志。
2. 增加cjson.c文件,已经在main.c中
3. 准备打包如下的JSON包
4. 代码部分,先将几个部分初始化指针
5. 创建链表
6. 我们查看添加的是否正确,可以将链表通过串口打印出来。
7. 效果如下:
四、解包
五、总结
(* ̄︶ ̄)创作不易!期待你们的 点赞、收藏和评论喔。
前言
在使用MQTT传输数据的时候,经常使用到JSON包,来进行数据的交互,这篇文章讲一下该如何使用。
一、JSON和cJson
- JSON:全称 JavaScript Object Notation,轻量级的数据格式。
- cJSON:是一个用c语言编写的JSON数据解析器,超级轻便,可以移植,单文件MIT源协议。
cJSON的源码文件只有两个:
cJSON.c
cJSON.h
下载地址可以在github下载,地址如下:
GitHub - DaveGamble/cJSON: Ultralightweight JSON parser in ANSI C
将这两个文件下载之后,添加到项目中,其它.C文件调用的时候,需要包含头文件cJSON.h即可,文件在MDK中的位置如下:
二、cJSON是如何表示JSON数据的
cJSON是用一个结构体来表示一个JSON数据,定义在cJSON.h中,查看如下源码:
/* The cJSON structure: */
typedef struct cJSON
{
/* next/prev allow you to walk array/object chains. Alternatively, use GetArraySize/GetArrayItem/GetObjectItem */
struct cJSON *next;
struct cJSON *prev;
/* An array or object item will have a child pointer pointing to a chain of the items in the array/object. */
struct cJSON *child;
/* The type of the item, as above. */
int type;
/* The item's string, if type==cJSON_String and type == cJSON_Raw */
char *valuestring;
/* writing to valueint is DEPRECATED, use cJSON_SetNumberValue instead */
int valueint;
/* The item's number, if type==cJSON_Number */
double valuedouble;
/* The item's name string, if this item is the child of, or is in the list of subitems of an object. */
char *string;
} cJSON;
不是存贮整片的JSON数据,而是按照每条数据进行存储的。使用next指针和prev指针进行链表,存储整片的JSON数据。
三、如何封装完整的JSON数据
说了这么多,我们实际操作一下,找一块STM32F开发板,测试一下。
1. 先将串口打通,方便电脑查看log日志。
uart_init(115200);
2. 增加cjson.c文件,已经在main.c中
#include "cJSON.h"
3. 准备打包如下的JSON包
{
"我的名字": "老刘",
"公众号": "数独机",
"年龄": 11,
"性别": "男",
"电话": 186,
"地址": "一般人不告诉",
"我是嵌套的cJSON": {
"号码": 9527,
"传真": 5510,
"淘宝店": "This熏"
},
"我是嵌套的数组": [{
"我是奇数循环": 0
}, {
"我是偶数循环": "12365"
}, {
"我是奇数循环": 2
}, {
"我是偶数循环": "12365"
}, {
"我是奇数循环": 4
}]
}
4. 代码部分,先将几个部分初始化指针
cJSON* main_root = NULL;
cJSON* main_m_root = NULL;
cJSON* main_l_root = NULL;
cJSON* obj=NULL;
char* response_str = NULL;
5. 创建链表
main_root = cJSON_CreateObject();
豪横的在链表中添加节点
cJSON_AddStringToObject(main_root, "我的名字", "老刘");
cJSON_AddStringToObject(main_root, "公众号", "数独机");
cJSON_AddNumberToObject(main_root, "年龄", 11);
cJSON_AddStringToObject(main_root, "性别", "男");
cJSON_AddNumberToObject(main_root, "电话", 186);
cJSON_AddStringToObject(main_root, "地址", "一般人不告诉");
main_m_root = cJSON_CreateObject();
cJSON_AddNumberToObject(main_m_root, "号码", 9527);
cJSON_AddNumberToObject(main_m_root, "传真", 5510);
cJSON_AddStringToObject(main_m_root, "淘宝店", "This熏");
cJSON_AddItemToObject(main_root, "我是嵌套的cJSON", main_m_root);
main_l_root=cJSON_CreateArray();
cJSON_AddItemToObject(main_root,"我是嵌套的数组",main_l_root);
for(i=0;i<5;i++)
{
obj=cJSON_CreateObject();
cJSON_AddItemToArray(main_l_root,obj);
if(i%2 == 0) cJSON_AddNumberToObject(obj, "我是奇数循环", i);
else cJSON_AddStringToObject(obj, "我是偶数循环", "12365");
}
6. 我们查看添加的是否正确,可以将链表通过串口打印出来。
response_str = cJSON_Print(main_root);
printf("%s\n",response_str);
7. 效果如下:
四、解包
解包的函数只有一个:
CJSON_PUBLIC(cJSON *) cJSON_Parse(const char *value);
还是用例子进行讲解:
将串口的数据完全接收到一个大的缓冲区里面,当然这个数据也可以来自wifi或者网络。
if(GET_STOP_flag)
{
for(i=0;i<(cJSON_pack_StopP-cJSON_pack_FistP)+1;i++)
{
MessageTmp[i] =WIFI_RC_buf[cJSON_pack_FistP-1+i];
}
}
GET_STOP_flag //为停止接收标志;
cJSON_pack_StopP //停止指针位置
cJSON_pack_FistP //开始指针位置
MessageTmp //临时的数组
只需要对临时的数组进行操作即可。开辟内存空间,将数组转换为字符串。
MessageStrTmp = (char *)cJSON_malloc(sizeof(char) * (sizeof(MessageTmp) + 1));
strncpy(MessageStrTmp, MessageTmp, sizeof(MessageTmp));
使用解包函数
//接收解包过程
Thisxun_root = cJSON_Parse(MessageStrTmp);
Thisxun_root_type = cJSON_GetObjectItem(hiss_root,"type")->valuestring;
if(strcmp(Thisxun_root_type,TYPE_USER_INFO_SYNC)==0)
{
Thisxun_insDetail = cJSON_GetObjectItem(Thisxun_root,"insDetail");
userID = cJSON_GetObjectItem(Thisxun_insDetail,"userId")->valuestring;
password = cJSON_GetObjectItem(Thisxun_insDetail,"password")->valuestring;
}
五、总结
每次封包和解包结束之后,都要及时释放内存:
cJSON_Delete(Thisxun_root);
cJSON_free(MessageStrTmp);
对于单片机而言,需要修改Heap_Size