cJSON的使用方法

cJSON的使用方法首先,要介绍一下什么是json格式。一.JSON格式简述JSON(JavaScriptObjectNotation)是一种轻量级的数据交换格式。易于人阅读和编写,同时也易于机器解析和生成。它基于JavaScript(StandardECMA-2623rdEdition-December1999)的一个子集。JSON采用完全独立于语言的文本格式,但

大家好,又见面了,我是你们的朋友全栈君。

首先,要介绍一下什么是json格式。

 

一.JSON格式简述

 

 

JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式。易于人阅读和编写,同时也易于机器解析和生成。它基于JavaScript(Standard ECMA-262 3rd Edition – December 1999)的一个子集。 JSON采用完全独立于语言的文本格式,但是也使用了类似于C语言家族的习惯(包括C, C++, C#, Java, JavaScript, Perl, Python等)。这些特性使JSON成为理想的数据交换语言。(来自“开源中国”资料)。
 
cJSON从名字可知,整个项目都是以极标准的C来写的,意思说,可以跨各种平台使用了。
cJSON
是一个超轻巧,携带方便,单文件,简单的可以作为ANSI-C标准的JSON解析器。
 
cJSON 开源项目位置:点击打开链接
更加详细的解释和示例请查看 http://www.json.org/ 主页。
 
cJSON,目前来说,就只有两个文件,一个cJSON.c 一个cJSON.h文件。使用的时候,自己创建好一个main.c文件后,将头文件include进去。
如果是在linux pc上,请使用以下命令进行编译:
gcc  *.c  cJSON.c    -lm

记得编译时末尾链接libm库。

 
 
二.JSON结构体
熟悉使用cJSON库函数可从cJSON结构体入手,cJSON结构体如下所示
typedef struct cJSON {   struct cJSON *next,*prev;    struct cJSON *child;    int type;    char *valuestring;    int valueint;   double valuedouble;    char *string;   } cJSON;  

 

 

几点说明

1.cJOSN结构体为一个双向列表,并可通过child指针访问下一层。
2.type变量决定数据项类型(键的类型),数据项可以是字符串可以是整形,也可以是浮点型。如果是整形值的话可从valueint,如果是浮点型的话可从valuedouble取出,以此类推。
3.string可理解为节点的名称,综合此处的第2点可理解为“键”的名称。
 

cJSON作为Json格式的解析库,其主要功能无非就是构建和解析Json格式了,用途就是一端将要发送的数据已cjson形式封装,然后发送,另一端收到此数据后,还是按cjson形式解析,就得到想要的数据了。

 
三.封装成json形式
接下来直接通过几个例子代码,一一解析。
   
第一,创建json数据串。这数据串,可能是对象,也可能是数组,也可能是它们的各种组合,其中再加上一些键值对。有一点要先说明:它们的组合,符合父子继承格式--这也是json数据串的特点。
<1>创建一个对象,并向这个对象里添加字符串和整型键值:
#include<stdio.h>#include<stdlib.h>#include<string.h>#include"cJSON.h"int main(){        cJSON * usr;        cJSON *arry;        usr=cJSON_CreateObject();   //创建根数据对象        cJSON_AddStringToObject(usr,"name","fengxin");  //加入键值,加字符串        cJSON_AddStringToObject(usr,"passwd","123");        cJSON_AddNumberToObject(usr,"num",1);  //加整数                char *out = cJSON_Print(usr);   //将json形式打印成正常字符串形式        printf("%s\n",out);               // 释放内存          cJSON_Delete(usr);                  }

 

 

运行结果:
{	"name":	"fengxin",	"passwd":	"123",	"num":	1}

 

若干说明

1. cJSON_CreateObject函数可创建一个根数据项,之后便可向该根数据项中添加string或int等内容,返回的是一个 cJSON的指针,注意,在这个指针用完了以后,需要手工调用 cJSON_Delete(root); 进行内存回收。
2. cJSON_AddNumberToObject向节点中添加子节点,例如此处添加name节点,节点值为字符串”fengxin”
3. 需要注意的是  json 格式的数据,虽然也是一个字符串的样子,但这个时候还是无法当成普通的字符串进行使用,需要调用 cJSON_PrintUnformatted(root) 或者 cJSON_Print(root);来将json对象转换成普通的字符串,并且都是以该json对象的根为基点。两个API的区别即是:一个是没有格式的:也就是转换出的字符串中间不会有”\n” “\t”之类的东西存在,而cJSON_Print(root);打印出来是人看起来很舒服的格式。
4. 因为函数内部封装有malloc函数,所以使用free函数释放被out占用的内存空间
<2> 创建一个数组,并向数组添加一个字符串和一个数字:
int create_js(void)
{
    cJSON *root, *js_body;
    root = cJSON_CreateArray();
    cJSON_AddItemToArray(root, cJSON_CreateString("Hello world"));
    cJSON_AddItemToArray(root, cJSON_CreateNumber(10)); 
    {
//      char *s = cJSON_Print(root);
        char *s = cJSON_PrintUnformatted(root);
        if(s){
           printf(" %s \n",s);
            free(s);
        }
    }
    if(root)
    cJSON_Delete(root);

    return 0;

}

int main(int argc, char **argv)
{
    create_js();
    return 0;
}

 

 
运行结果:
["Hello world",10]

<3> 对象里面包括一个数组,数组里面包括对象,对象里面再添加一个字符串和一个数字:

int create_js(void){    cJSON *root, *js_body, *js_list;    root = cJSON_CreateObject();    cJSON_AddItemToObject(root,"body", js_body = cJSON_CreateArray());    cJSON_AddItemToArray(js_body, js_list = cJSON_CreateObject());    cJSON_AddStringToObject(js_list,"name","fengxin");    cJSON_AddNumberToObject(js_list,"status",100);    {        //        char *s = cJSON_Print(root);        char *s = cJSON_PrintUnformatted(root);        if(s){            printf(" %s \n",s);            free(s);        }    }    if(root)        cJSON_Delete(root);    return 0;}int main(int argc, char **argv){    create_js();    return 0;}

 

运行结果:
{"body":[{"name":"fengxin","status":100}]}

<4>其他组合就依次类推,只要搞清楚父子关系即可。随便嵌套随便玩。不再贴了。

 
 
四.解析json得到数据
解析数据包的过程和组装数据包的过程相反

处理流程:

1, 先将普通的json串处理成json对象,也就是所谓的创建json root的过程,只有一行代码即可:

 

cJSON *root;
root = cJSON_Parse(js_string);

 

 

2,开始拿关键字,但如果关键字还有父层或者祖层,那就需要先从父层开拿,所谓剥洋葱是也!

先说没有父层的:

out={\"name\":\"fengxin\",\"passwd\":\"123\",\"num\":1}

 

 

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include"cJSON.h"

int main()
{
    cJSON *json,*json_name,*json_passwd,*json_num;
    char* out="{\"name\":\"fengxin\",\"passwd\":\"123\",\"num\":1}";

    json = cJSON_Parse(out); //解析成json形式
    json_name = cJSON_GetObjectItem( json , "name" );  //获取键值内容
    json_passwd = cJSON_GetObjectItem( json , "passwd" );
    json_num = cJSON_GetObjectItem( json , "num" );

    printf("name:%s,passwd:%s,num:%d\n",json_name->valuestring,json_passwd->valuestring,json_num->valueint);

    cJSON_Delete(json);  //释放内存 
    free(out);



}

 

 

 

显示结果:

 

name:fengxin,passwd:123,num:1

 

 

需要注意的是: 上面的type 已经在cJSON.h里面定义好了,有自己的意义。如果是在严格的场所,应该先判定该 item的type,然后再考虑去拿值。

而如果有父层的话,无非就是接着向下拿就是了

3.处理这串数据:

out={\"list\":{\"name\":\"xiao hong\",\"age\":10},\"other\":{\"name\":\"hua hua\"}}
 
char *s = "{\"list\":{\"name\":\"xiao hong\",\"age\":10},\"other\":{\"name\":\"hua hua\"}}";cJSON *root = cJSON_Parse(s);if(!root) {    printf("get root faild !\n");    return -1;}cJSON *js_list = cJSON_GetObjectItem(root, "list");if(!js_list) {    printf("no list!\n");    return -1;}printf("list type is %d\n",js_list->type);cJSON *name = cJSON_GetObjectItem(js_list, "name");if(!name) {    printf("No name !\n");    return -1;}printf("name type is %d\n",name->type);printf("name is %s\n",name->valuestring);cJSON *age = cJSON_GetObjectItem(js_list, "age");if(!age) {    printf("no age!\n");    return -1;}printf("age type is %d\n", age->type);printf("age is %d\n",age->valueint);cJSON *js_other = cJSON_GetObjectItem(root, "other");if(!js_other) {    printf("no list!\n");    return -1;}printf("list type is %d\n",js_other->type);cJSON *js_name = cJSON_GetObjectItem(js_other, "name");if(!js_name) {    printf("No name !\n");    return -1;}printf("name type is %d\n",js_name->type);printf("name is %s\n",js_name->valuestring);if(root)    cJSON_Delete(root);    return 0;

 

打印结果:

 

list type is 6
name type is 4
name is xiao hong
age type is 3
age is 10
list type is 6
name type is 4
name is hua hua

 

所谓子子孙孙无穷尽也,按照上面那个方法推下去即可。
 
4.json 里数组怎么取?
{\"list\":[\"name1\",\"name2\"]}
 
int main(int argc, char **argv){char *s = "{\"list\":[\"name1\",\"name2\"]}";cJSON *root = cJSON_Parse(s);if(!root) {    printf("get root faild !\n");    return -1;}cJSON *js_list = cJSON_GetObjectItem(root, "list");if(!js_list){    printf("no list!\n");    return -1;}int array_size = cJSON_GetArraySize(js_list);printf("array size is %d\n",array_size);int i = 0;cJSON *item;for(i=0; i< array_size; i++) {    item = cJSON_GetArrayItem(js_list, i);    printf("item type is %d\n",item->type);    printf("%s\n",item->valuestring);}if(root)    cJSON_Delete(root);    return 0;}

5.如果json数组里面又搞了对象怎么办? 

 

不怕搞对象,就怕这样搞对象? 无他,就是稍微复杂了一点,全是体力活儿:

<1> 既然是数组里面,那肯定要先测量数组的大小,然后根据大小循环拿;

<2> 拿到一个数组项,然后把这个项先转化成普通的json字符串,也就是 char *s 能接受的。

<3>再次将这个json字符串,转化成一个json对象

<4> 按照拿普通对象中的东西那样开干就行了。

 

 

{\"list\":[{\"name\":\"xiao hong\",\"age\":10},{\"name\":\"hua hua\",\"age\":11}]}

list 是一个数组,数组里面有两个对象,那么代码如下

 

 

int main(int argc, char **argv)
{
char *s = "{\"list\":[{\"name\":\"xiao hong\",\"age\":10},{\"name\":\"hua hua\",\"age\":11}]}";
cJSON *root = cJSON_Parse(s);
if(!root) {
    printf("get root faild !\n");
    return -1;
}
cJSON *js_list = cJSON_GetObjectItem(root, "list");
if(!js_list){
    printf("no list!\n");
    return -1;
}
int array_size = cJSON_GetArraySize(js_list);
printf("array size is %d\n",array_size);
int i = 0;
cJSON *item,*it, *js_name, *js_age;
char *p  = NULL;
for(i=0; i< array_size; i++) {
    item = cJSON_GetArrayItem(js_list, i);
    if(!item) {
        //TODO...
    }
    p = cJSON_PrintUnformatted(item);
    it = cJSON_Parse(p);
    if(!it)
        continue ;
    js_name = cJSON_GetObjectItem(it, "name");
    printf("name is %s\n",js_name->valuestring);
    js_age = cJSON_GetObjectItem(it, "age");
    printf("age is %d\n",js_age->valueint);

}

if(root)
    cJSON_Delete(root);
    return 0;
}

 

 

 

 

 

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/147802.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • Squid代理服务器(一)——大家所用的游戏代理软件到底为何物?

    Squid代理服务器(一)——大家所用的游戏代理软件到底为何物?一、代理服务器应用场景分析想当年大学时候宿舍控制网速,苦了我们这帮魔兽党,一到晚上工会活动我们就得和全校上万名学生抢网速,作为坦克的我总是因为网速问题导致团灭,咱也是有自尊的人,一怒之下花300元办

  • 电平转换的作用_电平转换电路原理

    电平转换的作用_电平转换电路原理作为一名电子设计的硬件工程师,电平转换是每个人都必须面对的的话题,主芯片引脚使用的1.2V、1.8V、3.3V等,连接外部接口芯片使用的1.8V、3.3V、5V等,由于电平不匹配就必须进行电平转换。每个工程师都有自己的一套转换方案,今天我们将5种电平转换的方法进行汇总,并且总结各种的优劣势,避免设计过程踩坑。一、电平转换方法5种电平转换方法分别是:晶体管电平转换方法;专用电平转换芯片;限流电阻电平转换方法;电阻分压电平转换方法;二极管电平转换方法;下面我们会从速率、驱动能力、漏电流、成本

  • matlab一阶惯性滤波,为什么一阶惯性环节也具有滤波的作用

    matlab一阶惯性滤波,为什么一阶惯性环节也具有滤波的作用在用simulink搭建模型的时候,发现一阶惯性环节具有滤波的作用,这是为什么呢?我们以一阶惯性环节200pi/(s+200pi)为例进行说明。首先从传递函数G(s)的频率特性说起。所谓系统的频率特性,是指系统在单位正弦相量作用下的稳态响应。因此,令传递函数中的s=jw,就可以得到系统的频率特性G(jw)。G(jw)是频率w的复变函数。他的幅值为|G(jw)|,相角为相角(G(jw))。当w从0到…

  • 5G LTE窄带物联网(NB-IoT) 10

    5G LTE窄带物联网(NB-IoT) 10第7章物理子层物理子层是底层子层,负责MACPDU的物理信道,传输和接收;如图7.1所示。RRC提供PHY子层的配置参数。在MAC/PHY接口,传输信道在发送和接收时分别映射到物理信道,反之亦然[28]。 RRC将其配置参数发送到每个子层,包括PHY子层,如第4.2,5.2,6.2和7.1节所示。7.1RRC配置参数RRC将专用或默认无线电配置参数发送到PHY子层,…

  • JAVA8 Collectors.groupingBy[通俗易懂]

    JAVA8 Collectors.groupingBy[通俗易懂]1.按长度对字符串进行分组List<String>list=Arrays.asList(“a”,”bb”,”cc”,”ddd”);Map<Integer,List<String>>result=list.stream().collect(Collectors.groupingBy(String::length));System.ou…

  • 鳄鱼mt4复盘助手_免费MT4复盘助手2015年8月新版

    鳄鱼mt4复盘助手_免费MT4复盘助手2015年8月新版本帖最后由宁静的魔于2015-8-2713:56编辑简介:鳄鱼复盘助手由鳄鱼金融独立研发,可对历史数据进行复盘测试,研究自己的交易系统。乃操盘交易必备神器。鳄鱼复盘助手完全免费,本论坛会员即可参与使用。本程序需要.netframework运行库,部分计算机没有装的下载安装一下,这个可以解决双击无反应.netframework4.0下载参与公测活动可得论坛纪念勋章,详情请点击:鳄…

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号