cJson
是一个用于解析json
文件的库,它由c语言编写,开源、使用简单。
cJSON git网站:https://github.com/DaveGamble/cJSON.
json 官网:http://www.json.org
下面是一个简单但覆盖了大多常用格式的json文件。后面会围绕它做简单的格式介绍。
{
"name": "guoweilkd",
"age": 18,
"high": 166.9,
"married": false,
"relation": {
"love": {
"name": "noName",
"loveTime": 6,
"age": 19
},
"interest": [
"篮球",
66.666,
19,
true,
{
"name": "noName",
"loveTime": 6,
"age": 19
}
]
}
}
由上面的json文件我们可以看到:
{}
包了起来。我们称它为root对象
。对象
,数组
,键值对
三种元素的各种嵌套组成。
键值对
使用"name":value
的格式。其中value
可以是下面的几种元素:
true
,false
.如"married":false
。"high":166.9
,"age":18
,"tmp":2E-3
."name":"noName"
。"relation":{}
。"interest":[]
。对象
使用{}
作为一个整体。数组
使用[]
作为一个整体。数组中的内容可以是下面几种元素的任意组合:
[{...},{...}]
。value
:如字符串,bool类型等。详见键值对value
的描述。,
分割。//
,/* */
的注释。对cJSON库最简单的使用是只包含cJSON.c
和cJSON.h
两个文件。其中所有的库接口都在cJSON.h
有详细描述。大体分为两部分:1.json文件的解析,2.生成json文件。后面我们结合代码详细讲解。
在谈cJSON库接口时,我们必须要先了解结构体struct cJSON
.了解了这个结构体于json文件之间的关系就相当于学会了如何使用此库。
typedef struct cJSON{
struct cJSON *next;//兄弟指针,指向后一个节点
struct cJSON *prev;//兄弟指针,指向前一个节点
struct cJSON *child;//儿子节点,指向子对象/数组
int type; //键值对中value值的类型。见键值对的描述
char *valuestring; //如果type为字符串flag,则此处为value的字符串值
int valueint; //如果type为数值flag,则此处为value的整型值
double valuedouble; //如果type为数值flag,则此处为value的浮点型值
char *string; //键值对的name名称。
} cJSON;
此程序生成的json文件为上面列举的json文件。可以对照学习。
#include
#include
#include "cJSON.h"
#include "string.h"
int main(void)
{
char *cjson_str = NULL;
cJSON *root = cJSON_CreateObject();
cJSON *relation = cJSON_CreateObject();
cJSON *love = cJSON_CreateObject();
cJSON *friend = cJSON_CreateObject();
cJSON *interest = cJSON_CreateArray();
/* 根节点下添加元素 */
cJSON_AddItemToObject(root, "name", cJSON_CreateString("guoweilkd"));//在对象中插入字符串
cJSON_AddItemToObject(root, "age", cJSON_CreateNumber(18)); //在对象中插入整型数值
cJSON_AddItemToObject(root, "high", cJSON_CreateNumber(166.9)); //在对象中插入浮点数值
cJSON_AddItemToObject(root, "married", cJSON_CreateFalse()); //在对象中插入bool量
/* 在root节点下添加relation子对象 */
cJSON_AddItemToObject(root, "relation", relation);
/* 在relation节点下添加love子对象 */
cJSON_AddItemToObject(relation, "love", love);
cJSON_AddItemToObject(love, "name", cJSON_CreateString("noName"));
cJSON_AddItemToObject(love, "loveTime", cJSON_CreateNumber(6));
cJSON_AddItemToObject(love, "age", cJSON_CreateNumber(19));
cJSON_AddItemToObject(friend, "name", cJSON_CreateString("noName"));
cJSON_AddItemToObject(friend, "loveTime", cJSON_CreateNumber(6));
cJSON_AddItemToObject(friend, "age", cJSON_CreateNumber(19));
/* 在relation节点下添加interest子对象 */
cJSON_AddItemToObject(relation, "interest", interest);
cJSON_AddItemToArray(interest,cJSON_CreateString("篮球")); //在数组中插入字符串
cJSON_AddItemToArray(interest,cJSON_CreateNumber(66.666)); //在数组中插入数值
cJSON_AddItemToArray(interest,cJSON_CreateNumber(19));
cJSON_AddItemToArray(interest,cJSON_CreateTrue()); //在数组中插入bool
cJSON_AddItemToArray(interest,friend); //在数组中插入对象
cjson_str = cJSON_Print(root); //将json转化为字符串,
/* 将json写入文件 */
FILE *pf = fopen("test.json","w");
fwrite(cjson_str,strlen(cjson_str),1,pf);
fclose(pf);
free(cjson_str); //释放cJSON_Print()函数申请的内存
cJSON_Delete(root); //删除一个cJSON实体和所有子实体
return 0;
}
下面是生成json文件过程中常用到的cJSON库接口:
cJSON_CreateNull(void)
:创建NULL类型cJSON_CreateTrue(void)
:创建bool->truecJSON_CreateFalse(void)
:创建bool->falsecJSON_CreateBool(cJSON_bool boolean)
创建bool(cJSON_True,cJSON_False)cJSON_CreateNumber(double num)
:创建整型或浮点型值。cJSON_CreateString(const char *string)
:创建字符串cJSON_CreateArray(void)
:创建数组cJSON_CreateObject(void)
:创建对象cJSON_AddItemToArray(cJSON *array, cJSON *item)
cJSON_AddItemToObject(cJSON *object, const char *string, cJSON *item)
cJSON_DetachItemFromArray(cJSON *array, int which)
cJSON_DeleteItemFromArray(cJSON *array, int which)
cJSON_DetachItemFromObject(cJSON *object, const char *string)
cJSON_DeleteItemFromObject(cJSON *object, const char *string)
cJSON_CreateIntArray(const int *numbers, int count)
cJSON_CreateFloatArray(const float *numbers, int count)
cJSON_CreateDoubleArray(const double *numbers, int count)
cJSON_CreateStringArray(const char **strings, int count)
cJSON_Duplicate(const cJSON *item, cJSON_bool recurse)
要解析的json文件为上面的json文件,可以对照学习。
#include
#include
#include "cJSON.h"
#include
void main(void)
{
/* 获取文件大小 */
struct stat fileBuff;
stat("test.json",&fileBuff);
printf("file size:%ld\n",fileBuff.st_size);
/* 为解析json文件分配缓存 */
char *pBuff = (char *)malloc(fileBuff.st_size);
FILE *pf = fopen("test.json","r");
fread(pBuff,fileBuff.st_size,1,pf);//将文件读入缓存
fclose(pf);
printf("file content:\n");
//printf("%s\n",pBuff);
/* 解析json文件 */
cJSON *root = NULL;
root = cJSON_Parse(pBuff);
if(root == NULL){
printf("parse error:%s\n",cJSON_GetErrorPtr());
return;
}
/* 方式1. 通过属性名查找 */
cJSON *pName = cJSON_GetObjectItem(root, "name");
if(pName == NULL){
printf("error\n");
}
else if(pName->type == cJSON_String){
printf("%s\n",pName->valuestring);
}
/* 方式2. 按照链表的格式,遍历查找 */
for(cJSON *pCur = root->child;pCur != NULL; pCur = pCur->next){
switch(pCur->type){
case cJSON_Invalid:break;
case cJSON_False :printf("%s = false\n",pCur->string);break;
case cJSON_True :printf("%s = true\n", pCur->string);break;
case cJSON_NULL :printf("%s = null\n", pCur->string);break;
case cJSON_Number :printf("%s = %d\n",pCur->string,pCur->valueint);break;
case cJSON_String :printf("%s = %s\n",pCur->string,pCur->valuestring);break;
case cJSON_Array :printf("%s = array\n",pCur->string);break;
case cJSON_Object :printf("%s = object\n",pCur->string);break;
default:printf("type:%d\n",pCur->type);break;
}
}
/* 获取数组大小和元素 */
cJSON *pTemp = cJSON_GetObjectItem(root, "relation");
cJSON *pInterest = cJSON_GetObjectItem(pTemp, "interest");
int ItemsNum = cJSON_GetArraySize(pInterest); //获取数组大小
printf("%s items is %d\n",pInterest->string,ItemsNum);
for(int i = 0; i < ItemsNum; i++){
cJSON *pCur = cJSON_GetArrayItem(pInterest,i);
switch(pCur->type){
case cJSON_Invalid:break;
case cJSON_False :printf("[%d] = false\n",i);break;
case cJSON_True :printf("[%d] = true\n",i);break;
case cJSON_NULL :printf("[%d] = null\n",i);break;
case cJSON_Number :printf("[%d] = %f\n",i,pCur->valuedouble);break;
case cJSON_String :printf("[%d] = %s\n",i,pCur->valuestring);break;
case cJSON_Array :printf("[%d] = array\n",i);break;
case cJSON_Object :printf("[%d] = object\n",i);break;
default:printf("type[%d]:%d\n",i,pCur->type);break;
}
}
/* 获取数组中对象的元素 */
pTemp = cJSON_GetArrayItem(pInterest,4);
cJSON *pt = cJSON_GetObjectItem(pTemp, "name");
printf("%s:%s\n",pt->string,pt->valuestring);
cJSON_Delete(root);
}
下面是解析json文件过程中常用到的cJSON库接口:
cJSON_Parse(const char *value)
cJSON_ParseWithOpts(const char *value, const char **return_parse_end, cJSON_bool require_null_terminated)
char *cJSON_GetErrorPtr(void)
:此函数会返回解析出错地方的指针。cJSON_GetArraySize(const cJSON *array)
:获取数组的大小。cJSON_GetArrayItem(const cJSON *array, int index)
:获取数组[index]
的对象指针。cJSON_GetObjectItem(const cJSON * const object, const char * const string)
char *cJSON_Print(const cJSON *item)
char *cJSON_PrintUnformatted(const cJSON *item)
cJSON_PrintBuffered(const cJSON *item, int prebuffer, cJSON_bool fmt)
cJSON_PrintPreallocated(cJSON *item, char *buffer, const int length, const cJSON_bool format)
cJSON_Delete(cJSON *c)
:释放一个cJSON实体和所有子实体分配的内存。const char*cJSON_Version(void)
:获取cjson库的版本号字符串。此处后的文字已经和题目内容无关,可以不看。
qq群:825695030
微信公众号:嵌入式的日常
如果上面的文章对你有用,欢迎打赏、点赞、评论。