查看原文
其他

一个超轻量级的JSON解析器

The following article is from 编程珠玑 Author 守望先生

来源:公众号【编程珠玑】

作者:守望先生

博客:https://www.yanbinghu.com


上一篇分享的基于Linux、C、JSON、Socket的编程实例(附代码)有用到cJSON,关于JSON是个什么东东之前也有写过JSON的简单认识关于cJSON是什么可以看这一篇:

前言

众所周知,JSON是一种轻量级的数据格式,应用广泛。在C/C++应用中也常常作为配置文件或者数据的存储,因此JSON文件的生成和解析是必备知识。

cJSON

cJSON是使用ANSI C编写的超轻量级的JSON解析器,因此在C中也常常是不二之选。
github 地址:https://github.com/DaveGamble/cJSON

下载到本地后,进行编译:

$ make

执行完成后即可在当前目录下得到libcjson.a和libcjson.so。

当然你也可以只下载cJSON.c和cJSON.h自己编译成静态库或动态库,可参考前期文章《如何制作静态库》和《动态库的制作和两种使用方式》。编译后的.a保留调试信息也只有不过43k。

关键数据结构

cJSON的关键数据结构如下:

typedef struct cJSON {  //cJSON结构体
       struct cJSON*next,*prev;           /*后驱节点和前驱节点*/
       struct cJSON *child;                   /*孩子节点*/
       int type;                                     /* 键的类型*/
       char *valuestring;                       /*字符串值*/
       int valueint;                                /* 整数值*/
       double valuedouble;                    /* 浮点数值*/
       char *string;                               /* 键的名字*/
} cJSON;

json是一种组织良好的数据格式,因而JSON中的内容解析后,都可以通过以上数据结构进行处理。
例如,对于下面的json内容:

{
    "name":"编程珠玑",
    "site":"https://www.yanbinghu.com",
    "age":1
}

解析后,site将会是name的next节点,并且它的键类型是字符串。

常用接口函数

cJSON *cJSON_Parse(const char *value);

用于将字符串解析成json对象,若失败则返回NULL。

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

用于获取json对象中的某个节点,若失败,返回NULL,成功则返回该节点对象。

void   cJSON_Delete(cJSON *c);

用于释放json对象相关内存。

char *cJSON_Print(const cJSON *item);

用于将JSON对象转换成字符串,记得最后释放相关内存。

JSON文件解析准备

解析JSON文件可大致分为以下几个步骤:

  • 获取文件大小

  • 将JSON文件内容读取到buffer

  • 通过cJSON接口解析buffer中的字符串

  • 获取JSON指定字段

为了将JSON文件的内容读取到buffer,需要知道文件的大小:

//来源:公众号【编程珠玑】
//https://www.yanbinghu.com
size_t get_file_size(const char *filepath)
{
    /*check input para*/
    if(NULL == filepath)
        return 0;
    struct stat filestat;
    memset(&filestat,0,sizeof(struct stat));
    /*get file information*/
    if(0 == stat(filepath,&filestat))
        return filestat.st_size;
    else
        return 0;
}

然后申请一段内存,将文件中的文本读取到buffer中:

//来源:公众号【编程珠玑】
//https://www.yanbinghu.com
char *read_file_to_buf(const char *filepath)
{
    /*check input para*/
    if(NULL == filepath)
    {
        return NULL;
    }
    /*get file size*/
    size_t size = get_file_size(filepath);
    if(0 == size)
        return NULL;

    /*malloc memory*/
    char *buf = malloc(size+1);
    if(NULL == buf)
        return NULL;
    memset(buf,0,size+1);

    /*read string from file*/
    FILE *fp = fopen(filepath,"r");
    size_t readSize = fread(buf,1,size,fp);
    if(readSize != size)
    {
        /*read error*/
        free(buf);
        buf = NULL;
    }

    buf[size] = 0;
    return buf;
}

再根据前面提到的解析流程,我们的JSON预解析函数如下:

cJSON *prepare_parse_json(const char *filePath)
{
    /*check input para*/
    if(NULL == filePath)
    {
        printf("input para is NULL\n");
        return NULL;
    }
    /*read file content to buffer*/
    char *buf = read_file_to_buf(filePath);
    if(NULL == buf)
    {
        printf("read file to buf failed\n");
        return NULL;
    }
    /*parse JSON*/
    cJSON *pTemp = cJSON_Parse(buf);
    free(buf);
    buf = NULL;
    return pTemp;
}

来源:公众号【编程珠玑】
网站:https://www.yanbinghu.com

解析JSON文件

假设我们的JSON文件内容如下:

{
    "name":"编程珠玑",
    "site":"https://www.yanbinghu.com",
    "age":1
}

按照我们前面梳理的解析JSON的步骤,main函数代码如下:

int main(void)
{
    char *filename = "./test.json";
    cJSON *pJson = NULL;
    cJSON *pTemp = NULL;
    pJson = prepare_parse_json(filename);
    if(NULL == pJson)
    {
        printf("parse json failed\n");
        return -1;
    }
    /*获取name值*/
    pTemp = cJSON_GetObjectItem(pJson,"name");
    printf("name is %s\n",pTemp->valuestring);
    /*获取site值*/
    pTemp = cJSON_GetObjectItem(pJson,"site");
    printf("site is %s\n",pTemp->valuestring);

    /*获取age值*/
    pTemp = cJSON_GetObjectItem(pJson,"age");
    printf("age is %d\n",pTemp->valueint);
    /*记得释放相关内存*/
    cJSON_Delete(pJson);
    pJson = NULL;
    return 0;
}

从上面看来,我们自己封装好之后,解析json似乎也没有那么复杂?
编译:

$ gcc -L .  -o parseJson parseJson.c -lcjson

注意指定链接cjson库的路径。

运行:

$ ./parseJson
name is 编程珠玑
site is https://www.yanbinghu.com
age is 1

写JSON

写JSON又该如何呢?实际上不过是组装JSON对象罢了。示例代码如下:

#include<stdio.h>
#include<stdlib.h>
#include"cJSON.h"
int main(void)
{
    cJSON *pRoot = NULL;
    char *json = NULL;
    cJSON *array = NULL;
    //创建根节点对象
    pRoot = cJSON_CreateObject();
    //向根节点加入数字对象
    cJSON_AddNumberToObject(pRoot, "age"1);
    //向根节点加入字符串对象
    cJSON_AddStringToObject(pRoot, "name""编程珠玑");
    //创建数组对象
    array = cJSON_CreateArray();
    cJSON_AddStringToObject(array,"language","C");
    cJSON_AddStringToObject(array,"language","C++");
    //向根节点中添加数组
    cJSON_AddItemToObject(pRoot,"array",array);
    //解析成字符串
    json = cJSON_Print(pRoot);
    printf("%s\n", json);//这里也可以将字符串写入文件
    //释放json对象的空间
    cJSON_Delete(pRoot);
    pRoot = NULL;
    //记得释放json的空间
    free(json);
    json = NULL;
    return 0;
}

运行结果:

{
    "age":  1,
    "name""编程珠玑",
    "array":    ["C""C++"]
}

总结

相比于python一条搞定解析来说,C语言中解析JSON似乎显得有些麻烦,但cJSON无疑是一个超轻量级的JSON器。

本文完整代码和点击阅读原文或者访问
http://www.yanbinghu.com/2019/08/04/21364.html
查看附录部分。

猜你喜欢:

基于Linux、C、JSON、Socket的编程实例(附代码)

教你写一个定时关机程序



    您可能也对以下帖子感兴趣

    文章有问题?点此查看未经处理的缓存