【常见开源库的二次开发】一文学懂CJSON

简介:

        JSON(JavaScript Object Notation)是一种轻量级的数据交换格式。它基于JavaScript的一个子集,但是JSON是独立于语言的,这意味着尽管JSON是由JavaScript语法衍生出来的,它可以被任何编程语言读取和生成。JSON的设计目的是使数据交换既简单又快速,相对于其他数据交换格式如XML,JSON更加轻巧,这使得它在网络上传输更加高效。对于人类和机器来说,JSON文本的格式都是易于理解的。 管JSON源于JavaScript,但是几乎所有的编程语言都有解析JSON的库,这使得JSON成为跨平台和语言的数据交换的理想格式。

 一、什么是json? 

        JSON(JavaScript Object Notation)是一种轻量级的数据交换格式。它基于JavaScript的一个子集,但是JSON是独立于语言的,这意味着尽管JSON是由JavaScript语法衍生出来的,它可以被任何编程语言读取和生成。JSON的设计目的是使数据交换既简单又快速。

1.1 JSON的特点:

         1. 轻量级:相对于其他数据交换格式如XML,JSON更加轻巧,这使得它在网络上传输更加高效。

        2. 易于阅读和编写:对于人类和机器来说,JSON文本的格式都是易于理解的。

        3. 语言无关性:尽管JSON源于JavaScript,但是几乎所有的编程语言都有解析JSON的库,这使得JSON成为跨平台和语言的数据交换的理想格式。

1.2 JSON的结构:

        1. 键值对集合(在其他语言中可能被实现为对象,记录,结构,字典,哈希表,有名列表,或者关联数组)。在JSON中,它们被表示为一个由花括号包围的对象。每个键值对由一个键(字符串)和一个值组成,键值之间用冒号分隔,键值对之间用逗号分隔。

        2. 有序的值列表(在大多数语言中被实现为数组)。在JSON中,它们被表示为由方括号包围的数组,数组的元素之间用逗号分隔。

一个简单的JSON对象示例:

{"name": "John Doe","age": 30,"isEmployed": true,"address": {"street": "123 Main St","city": "Anytown"},"phoneNumbers": ["123-456-7890","987-654-3210"]
}

        在这个例子中,我们有一个包含五个键值对的JSON对象。`name`、`age`和`isEmployed`键对应的值是简单的数据类型(字符串、数字和布尔值)。`address`键对应的值是一个嵌套的JSON对象,而`phoneNumbers`键对应的值是一个包含字符串的JSON数组。

1.3 基本元素

JSON(JavaScript Object Notation)的语法规则相对简单,主要包括以下几个基本元素:

        1.对象(Object):用花括号 `{}` 表示,包含一组无序的键值对。每个键值对之间用逗号分隔。键必须是字符串,值可以是任意有效的 JSON 数据类型。

     {"key1": "value1","key2": "value2","key3": "value3"}

        2. 数组(Array):用方括号 `[]` 表示,包含一组有序的值。每个值之间用逗号分隔。数组中的值可以是任意有效的 JSON 数据类型。

    {"arrayKey": [1, 2, 3, 4]}

        3. 值(Value):可以是字符串、数字、布尔值、对象、数组或 `null`。这些值可以嵌套在对象或数组中。

    {"stringKey": "Hello, JSON!","numberKey": 42,"booleanKey": true,"nullKey": null,"objectKey": {"nestedKey": "nestedValue"},"arrayKey": [1, "two", false, null]}

        4. 字符串(String): 字符串在JSON中是由双引号 `" "` 包围的一系列Unicode字符。字符串用于表示文本数据。

{"name": "John Doe","city": "New York"}

        5. 数字(Number): 数字可以是整数或者浮点数,直接写出,不需要加引号。JSON中的数字和大多数编程语言中的表示方法相似。

        {"integer": 12,"float": 3.14}

        6. 布尔值(Boolean): 布尔值表示逻辑实体,只有两个值,真(`true`)或假(`false`),不需要加引号。

        {"isTrue": true,"isFalse": false}

        7. null: `null` 在JSON中表示空值或不存在的值。它没有引号。

        {"emptyValue": null}

具体示例:

{"name": "John Doe","age": 30,"isEmployed": true,"address": {"street": "123 Main St","city": "Anytown"},"phoneNumbers": ["123-456-7890","987-654-3210"]
}

在这个示例中:

        name、age 和 isEmployed 是简单的键值对,值分别是字符串、数字和布尔值。        

        address 是一个嵌套的对象,包含 street 和 city 两个键值对。

        phoneNumbers  是一个数组,包含两个字符串值。

        JSON 的简洁和易读特性使它成为数据交换和配置文件的理想选择。无论是前后端数据传递还是存储配置,JSON 都非常适用。

一个更加复杂的json:t.weather.itboy.net/api/weather/city/101010100

1.4 使用场景:

        Web开发:JSON广泛用于前后端之间的数据交换。

        配置文件:许多应用程序使用JSON格式来存储配置设置。

        API和Web服务:许多Web服务使用JSON格式来提供公共API,因为它易于被不同的编程语言读取和解析。

由于其简洁、易于理解的结构,JSON已成为Web应用和服务之间交换数据的事实标准。

 二、下载json 

下载链接:GitHub - DaveGamble/cJSON: Ultralightweight JSON parser in ANSI C

解压后得到这俩个文件 

我们需要将这俩个文件加入到vscode中的C程序文件夹中

三、创建一个json

以下是一个使用 cJSON 创建一个JSON对象的示例代码:

#include <stdio.h>
#include <stdlib.h>
#include "cJSON.h"
int main()
{// 创建一个新的JSON对象cJSON *root = cJSON_CreateObject();// 向根对象中添加一个名为"key1"的键值对,值为字符串"10"cJSON* key1 = cJSON_AddStringToObject(root,"key1","10");if(key1 != NULL){printf("key:%s value :%s\n",key1->string,key1->valuestring);}// 将JSON对象转换为字符串格式char *str = cJSON_Print(root);  //输出json字符串// 打印转换后的JSON字符串printf(str);// 释放转换成字符串所使用的内存free(str);// 暂停程序运行,以便查看控制台输出system("pause>0");system("pause>0");return 0;
}

        在这个代码中,我们创建了一个空的JSON对象 root 

3.1 创建json对象 

    // 创建一个新的JSON对象cJSON *root = cJSON_CreateObject();

        cJSON_CreateObject 函数,这是 cJSON 库中用来创建一个新的JSON对象的函数。cJSON 是一个在C语言中处理JSON数据的轻量级库。

如果您想使用这个库来创建一个JSON对象,您可以按照以下步骤进行:

        1. 首先,确保已经安装了 cJSON 库。

        2. 在C语言代码中,包含 cJSON 的头文件。

        3. 使用 cJSON_CreateObject 函数来创建一个新的JSON对象。

 3.2 cjson的类型详解

cjson结构体如下定义:

/* cJSON 结构体: */
typedef struct cJSON
{/* next/prev 允许你在数组/对象链中遍历。或者,使用 GetArraySize/GetArrayItem/GetObjectItem */struct cJSON *next;struct cJSON *prev;/* 数组或对象项将有一个子指针,指向数组/对象中的项链。 */struct cJSON *child;/* 项的类型,如上所述。 */int type;/* 项的字符串,如果 type==cJSON_String 和 type == cJSON_Raw */char *valuestring;/* 写入 valueint 已被弃用,请使用 cJSON_SetNumberValue 代替 */int valueint;/* 项的数字,如果 type==cJSON_Number */double valuedouble;/* 项的名称字符串,如果此项是对象的子项,或者在对象的子项列表中。 */char *string;
} cJSON;

        这段代码定义了一个名为 cJSON 的结构体,这是 cJSON 库的核心数据结构,它表示一个JSON数据项。每一个 cJSON 结构体可以表示一个JSON对象、数组、字符串、数字或其他的JSON数据类型。

下面是每个成员变量的功能:

        struct cJSON *next;` 和 `struct cJSON *prev;:这两个指针分别指向相邻的JSON数据项,它们使得 cJSON 结构体可以形成一个链表,这对于表示一个JSON数组或对象是十分有用的。

        struct cJSON *child;:如果当前 cJSON 结构体表示一个JSON数组或对象,child 指针会指向一个链表,这个链表的元素就是数组或对象中的数据项。

        int type;:这个变量表示当前 cJSON 结构体表示的JSON数据类型,例如,它可能是 cJSON_NULL、cJSON_Number、cJSON_String、cJSON_Array 或 cJSON_Object 等。

        char *valuestring;:如果当前 cJSON 结构体表示一个JSON字符串,valuestring 就会被设置成那个字符串的值。如果它表示一个原始的未解析的JSON字符串,valuestring 也会被设置成那个字符串的值。

        int valueint;:这个成员现在已经不建议使用,取而代之的是 cJSON_SetNumberValue 函数来设置JSON数字的值

        double valuedouble;:如果当前 cJSON 结构体表示一个JSON数字,valuedouble 就会被设置成那个数字的值。

        char *string;:如果当前 cJSON 结构体是一个JSON对象中的数据项,string 就会被设置成那个数据项对应的键名

        我们可以使用 `cJSON` 库提供的各种函数,如 cJSON_CreateObject、cJSON_AddItemToObject 或 cJSON_GetObjectItem等,来创建、操作和查询 cJSON 结构体。

3.3 创建键值对

        cJSON_AddStringToObject 。这个函数是 cJSON 库中的一个函数,用于向一个 JSON 对象中添加一个字符串类型的键值对。

cJSON *cJSON_AddStringToObject(cJSON * const object, const char * const name, const char * const string);

        cJSON * const object:这是一个指向 JSON 对象的指针,表示你要往哪个对象中添加键值对。
        const char * const name:这是一个字符串,表示你要添加的键的名称(key)。

        const char * const string:这是一个字符串,表示你要添加的值(value)。

返回值:

        cJSON :这个函数返回一个指向新添加的 JSON 元素的指针。这个元素包含了添加的字符串值。如果添加失败,返回 NULL。

3.4 添加嵌套的JSON对象

        cJSON_AddItemToObject,来自 cJSON 库。函数原型应该是:

cJSON_bool cJSON_AddItemToObject(cJSON *object, const char *string, cJSON *item);

        这个函数的作用是将一个 cJSON 结构(`item`),它可以代表任何类型的JSON数据(如对象、数组、字符串、数字等),添加到另一个 cJSON 对象结构中,并将其与指定的键(`string`)相关联。

参数解释如下:

        cJSON *object:这是一个指向目标JSON对象的指针,你会向它添加一个新的元素。

        const char *string:这是你想要在目标对象中创建的键的名称。

        cJSON *item:这是一个指向你想要添加的 `cJSON` 元素的指针。

返回值:

        cJSON_bool:这是一个布尔值,如果成功添加元素则返回 `cJSON_True`,否则返回 `cJSON_False`。

下面是一个使用 `cJSON_AddItemToObject` 函数添加嵌套 JSON 对象的示例:

#include <stdio.h>
#include <stdlib.h>
#include "cJSON.h"
int main()
{// 创建一个新的JSON对象cJSON *root = cJSON_CreateObject();// 向根对象中添加一个名为"key1"的键值对,值为字符串"10"cJSON* key1 = cJSON_AddStringToObject(root,"key1","10");cJSON* last = NULL; // 用于保存上一个添加的键值对for(int i = 0; i < 5; i ++){// 创建一个名为"obj1"的子对象cJSON* obj1 = cJSON_CreateObject();// 将子对象添加到根对象中cJSON_AddItemToObject(root,"obj1",obj1);     // 向子对象中添加一个名为"key2"的键值对,值为字符串"20"cJSON* key2 = cJSON_AddStringToObject(obj1,"key2","20"); // 将子对象添加到根对象中last=key2;}if(key1 != NULL){printf("key:%s value :%s\n",key1->string,key1->valuestring);}// 将JSON对象转换为字符串格式char *str = cJSON_Print(root);  //输出json字符串// 打印转换后的JSON字符串printf(str);// 释放转换成字符串所使用的内存free(str);// 暂停程序运行,以便查看控制台输出system("pause>0");system("pause>0");return 0;
}

        在这个示例中,我们创建了一个根对象 root 和一个嵌套对象 nested,然后向嵌套对象添加了一个字符串键值对,并将嵌套对象作为根对象的一个字段添加进去。最后,我们打印并释放了相关的内存。

 四、添加数组

4.1 创建数组

cSON_PUBLIC(cSON *) cSON_CreateArray(void);

这个函数声明创建一个空的JSON数组,并返回一个指向该数组的`cJSON`指针

`cJSON_PUBLIC`是一个宏,用于定义函数的可见性,通常用于导出函数供外部使用。

函数没有参数,因此调用时不需要传递任何参数。返回值是一个指向新创建的JSON数组的指针。你可以使用这个指针来添加元素到数组中,或者将数组添加到其他JSON对象中。

示例用法:

cJSON *array = cJSON_CreateArray();
if (array == NULL) {fprintf(stderr, "错误: 无法创建JSON数组。\n");return 1;
}// 现在你可以使用 array 指针来操作这个JSON数组

确保在使用完数组后调用`cJSON_Delete(array)`来释放内存,以避免内存泄漏。

4.2 添加元素到数组

cJSON_PUBLIC(cJSON_bool) cJSON_AddItemToArray(cJSON *array, cJSON *item);

这个函数的作用是将一个JSON元素添加到指定的JSON数组中。函数的返回值是一个布尔值,表示添加操作是否成功。如果成功添加元素,返回true,否则返回false

参数说明:

  • cJSON *array: 这是一个指向JSON数组的指针,表示你要往哪个数组中添加元素。
  • cJSON *item: 这是一个指向要添加的JSON元素的指针。可以是任何JSON数据类型,比如字符串、数字、对象等。

我们首先创建了一个空的JSON数组和一个字符串元素,然后将字符串元素添加到数组中。如果添加操作成功,数组将包含这个新添加的元素。如果任何步骤失败,我们会打印错误信息并释放已分配的内存。

#include <stdio.h>
#include <stdlib.h>
#include "cJSON.h"int main()
{// 创建一个JSON对象cJSON *root = cJSON_CreateObject();if (root == NULL) {fprintf(stderr, "错误: 无法创建JSON对象。\n");return 1;}// 定义一个整数数组int a[10] = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10};// 创建一个包含整数数组的JSON数组cJSON *array = cJSON_CreateIntArray(a, 10);if (array == NULL) {fprintf(stderr, "错误: 无法创建JSON数组。\n");cJSON_Delete(root);return 1;}// 将JSON数组添加到JSON对象中,键名为"array"cJSON_AddItemToObject(root, "array", array);// 将JSON对象转换为字符串并打印char *json_str = cJSON_Print(root);if (json_str != NULL) {printf("生成的JSON对象: \n%s\n", json_str);free(json_str);} else {fprintf(stderr, "错误: 无法打印JSON对象。\n");}// 释放JSON对象cJSON_Delete(root);// 暂停程序运行,以便查看控制台输出system("pause");return 0;
}
  1. 创建一个JSON对象。
  2. 定义一个整数数组。
  3. 创建一个包含整数数组的JSON数组。
  4. 将JSON数组添加到JSON对象中,键名为"array"。
  5. 将JSON对象转换为字符串并打印。
  6. 释放JSON对象。
  7. 暂停程序运行,以便查看控制台输出。

五、将JSON对象转为字符串

cSON_Print 函数的作用是将一个 cJSON 元素转换成字符串,并返回这个字符串的指针。这个函数是用于将 JSON 对象或数组转换为可读的字符串格式。

CSON_PUBLIC(char *) cSON_Print(const cSON *item);
#include <stdio.h>
#include <stdlib.h>
#include "cJSON.h"int main()
{// 创建一个JSON对象cJSON *root = cJSON_CreateObject();if (root == NULL) {fprintf(stderr, "错误: 无法创建JSON对象。\n");return 1;}// 添加一些数据到JSON对象中cJSON_AddNumberToObject(root, "number", 123);cJSON_AddStringToObject(root, "string", "Hello, World!");// 将JSON对象转换为字符串char *json_str = cJSON_Print(root);if (json_str != NULL) {printf("生成的JSON字符串: \n%s\n", json_str);// 释放动态分配的字符串内存free(json_str);} else {fprintf(stderr, "错误: 无法打印JSON对象。\n");}// 释放JSON对象cJSON_Delete(root);return 0;
}

我们首先创建了一个 JSON 对象 root,并向其中添加了一个数字和一个字符串。然后,我们使用 cJSON_Print 函数将这个 JSON 对象转换为字符串,并打印出来。最后,我们释放了动态分配的字符串内存和 JSON 对象。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/43981.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

01 | 基础架构:一条SQL查询语句是如何执行的?

此系列文章为极客时间课程《MySQL 实战 45 讲》的学习笔记&#xff01; 引言 在了解 SQL 查询语句如何执行之前&#xff0c;先了解下MySQL 的基本架构示意图。 MySQL 分为 Server 层和引擎层。 Server 层包括连接器、查询缓存、分析器、优化器、执行器等&#xff0c;涵盖 M…

uniapp自动升级

一、创建云服务空间&#xff08;https://unicloud.dcloud.net.cn&#xff09; 云空间用于关联需要版本控制升级的项目&#xff0c;如果已拥有云空间则省略此步骤。 二、搭建 uni升级中心 - 后台管理系统&#xff08;升级中心 uni-upgrade-center - Admin&#xff09; uni-adm…

Apache部署与配置

概述 介绍 Apache HTTP Server(简称Apache)是Apache的一个开源的网页服务器&#xff0c;它源自NCSAhttpd服务器&#xff0c;并经过多次修改和发展&#xff0c;如今已经成为全球范围内广泛使用的Web服务器软件之一 特点 跨平台&#xff1a;可以运行在几乎所有广泛使用的计算机平…

Apache中使用SSI设置

先停服务在修改httpd.conf&#xff0c;备份下 Apache\Apache24\conf 设置httpd.conf LoadModule ssl_module modules/mod_ssl.so 取消该命令前的注释符# AddType text/html .shtml AddOutputFilter INCLUDES .shtml 取消该命令前的注释符# 加入html AddType text/html .…

前端javascript中的排序算法之插入排序

插入排序&#xff08;Selection Sort&#xff09;基本思想&#xff1a; 插入排序每次排一个数组项&#xff0c;以此方式构建最后的排序数组。假定第一项已经排序了&#xff0c;接着&#xff0c; 它和第二项进行比较&#xff0c;第二项是应该待在原位还是插到第一项之前呢&#…

软件工具网站推荐

1.菜鸟工具 菜鸟工具 - 不止于工具菜鸟工具&#xff0c;为开发设计人员提供在线工具&#xff0c;网址导航&#xff0c;提供在线PHP、Python、 CSS、JS 调试&#xff0c;中文简繁体转换&#xff0c;进制转换等工具。致力于打造国内专业WEB开发工具&#xff0c;集成开发环境&…

详细谈谈负载均衡的startupProbe探针、livenessProbe探针、readnessProbe探针如何使用以及使用差异化

文章目录 startupProbe探针startupProbe说明示例配置参数解释 使用场景说明实例——要求&#xff1a; 容器在8秒内完成启动&#xff0c;否则杀死对应容器工作流程说明timeoutSeconds: 和 periodSeconds: 参数顺序说明 livenessProbe探针livenessProbe说明示例配置参数解释 使用…

CSS技巧专栏:一日一例 1.纯CSS实现 会讨好的热情按钮 特效

题外话: 从今天开始,我准备开设一个新的专栏,专门写 使用CSS实现各种酷炫按钮的方法,本专栏目前准备写40篇左右,大概会完成如下按钮效果: 今天,我来介绍第一个按钮的实现方法:会讨好的热情按钮。为什么我给它起这样的名字呢?你看它像不像一个不停摇尾巴的小黄?当你鼠…

【QML之·基础语法概述】

系列文章目录 文章目录 前言一、QML基础语法二、属性三、脚本四、核心元素类型4.1 元素可以分为视觉元素和非视觉元素。4.2 Item4.2.1 几何属性(Geometry&#xff09;:4.2.2 布局处理:4.2.3 键处理&#xff1a;4.2.4 变换4.2.5 视觉4.2.6 状态定义 4.3 Rectangle4.3.1 颜色 4.4…

【界面态】霍尔效应表征氮化对SiC/SiO2界面陷阱的影响

引言 引言主要介绍了硅碳化物&#xff08;SiC&#xff09;金属-氧化物-半导体场效应晶体管&#xff08;MOSFETs&#xff09;作为新一代高压、低损耗功率器件的商业化背景。SiC MOSFETs因其优越的电气特性&#xff0c;在高电压和高温应用领域具有巨大的潜力。然而&#xff0c;尽…

综合安全防护

题目 1,DMZ区内的服务器,办公区仅能在办公时间内(9:00-18:00)可以访问,生产区的设备全天可以访问. 2,生产区不允许访问互联网,办公区和游客区允许访问互联网 3,办公区设备10.0.2.10不允许访问DMz区的FTP服务器和HTTP服务器,仅能ping通10.0.3.10 4,办公区分为市场部和研发部,研…

动态数据库设计

动态数据库设计是一种灵活的方法&#xff0c;用于构建能够适应不断变化的数据需求的数据库结构。它强调在不频繁修改数据库表结构的前提下&#xff0c;有效管理和存储多样化的数据。以下是实现动态数据库设计的一些关键技术点和策略&#xff1a; 实体-属性-值&#xff08;EAV&a…

如何在JetBrains中写Codeforce?

目录 前言 正文 leetcode 个人喜好 参考资料 具体操作步骤 尾声 &#x1f52d; Hi,I’m Pleasure1234&#x1f331; I’m currently learning Vue.js,SpringBoot,Computer Security and so on.&#x1f46f; I’m studying in University of Nottingham Ningbo China&#x1f4…

Python函数 之 模块和包

1.模块 1, 在Python 中, 每个以 .py 结尾的 Python 代码⽂件 都可以称为是⼀个模块。 2, 在模块中 别⼈书写好的功能(变量, 函数, 类)&#xff0c;我们可以拿来直接使⽤。 3, 我们自己写的代码文件&#xff0c; 想要作为模块让别⼈使⽤, 你的代码⽂件名(模块名) 满足标识符的规…

物流工业三防平板实时跟踪货物位置和状态

在当今全球化和高度数字化的商业环境中&#xff0c;物流行业的高效运作对于企业的成功和经济的繁荣至关重要。货物的准确、实时跟踪不仅能提高物流效率&#xff0c;还能增强客户满意度&#xff0c;降低运营成本。物流工业三防平板的出现&#xff0c;为实现货物位置和状态的实时…

全网最适合入门的面向对象编程教程:12 类和对象的 Python 实现-Python 使用 logging 模块输出程序运行日志

全网最适合入门的面向对象编程教程&#xff1a;12 类和对象的 Python 实现-Python 使用 logging 模块输出程序运行日志 摘要&#xff1a; 本文主要介绍了日志的定义和作用&#xff0c;以及 Python 内置日志处理的 logging 模块&#xff0c;同时简单说明了日志等级和 logging …

【人工智能】-- 搜索技术(状态空间法)

个人主页&#xff1a;欢迎来到 Papicatch的博客 课设专栏 &#xff1a;学生成绩管理系统 专业知识专栏&#xff1a; 专业知识 文章目录 &#x1f349;引言 &#x1f348;介绍 &#x1f349;状态空间法 &#x1f348;状态空间的构成 &#x1f34d;状态 &#x1f34d;算符…

办公助手推荐✨

办公助手来啦&#xff01;✨ 办公助手来啦&#xff01;✨&#x1f31f; 主要亮点&#x1f4dd; 全新PDF编辑器&#x1f3a8; 丰富的幻灯片版式&#x1f30d; 改进的从右至左显示&#x1f310; 新增本地化选项 &#x1f4ca; 应用场景在线办公套件&#x1f4f1; 多平台支持&…

idea创建dynamic web project

由于网课老师用的是eclipse,所以又得自己找教程了…… 解决方案&#xff1a; https://blog.csdn.net/Awt_FuDongLai/article/details/115523552

20240709每日后端--------最优解决Invalid bound statement (not found)

目标 最优解决Invalid bound statement (not found) 步骤 1、打包 2、查看target下是否成双成对出现 3、核对无误后&#xff0c;即可解决问题。