基于esp32-s3,写一个实现json键值对数据创建和读写解析c例程

以下是一个基于 ESP32 - S3 使用 ESP - IDF 框架实现 JSON 键值对数据创建、读写和解析的 C 语言例程。

环境准备

确保你已经安装了 ESP - IDF 开发环境,并且可以正常编译和烧录代码到 ESP32 - S3 开发板。

代码示例

#include <stdio.h>
#include <string.h>
#include "esp_log.h"
#include "cJSON.h"// 日志标签
static const char *TAG = "JSON_EXAMPLE";// 创建 JSON 键值对数据
char* create_json_data() {// 创建一个新的 JSON 对象cJSON *root = cJSON_CreateObject();if (root == NULL) {ESP_LOGE(TAG, "Failed to create JSON object");return NULL;}// 添加键值对到 JSON 对象cJSON_AddStringToObject(root, "name", "John Doe");cJSON_AddNumberToObject(root, "age", 30);cJSON_AddBoolToObject(root, "is_student", false);// 将 JSON 对象转换为字符串char *json_string = cJSON_Print(root);if (json_string == NULL) {ESP_LOGE(TAG, "Failed to print JSON object");cJSON_Delete(root);return NULL;}// 删除 JSON 对象cJSON_Delete(root);return json_string;
}// 解析 JSON 数据
void parse_json_data(const char *json_string) {// 解析 JSON 字符串cJSON *root = cJSON_Parse(json_string);if (root == NULL) {const char *error_ptr = cJSON_GetErrorPtr();if (error_ptr != NULL) {ESP_LOGE(TAG, "Error before: %s", error_ptr);}return;}// 读取键值对cJSON *name = cJSON_GetObjectItemCaseSensitive(root, "name");cJSON *age = cJSON_GetObjectItemCaseSensitive(root, "age");cJSON *is_student = cJSON_GetObjectItemCaseSensitive(root, "is_student");if (cJSON_IsString(name) && (name->valuestring != NULL)) {ESP_LOGI(TAG, "Name: %s", name->valuestring);}if (cJSON_IsNumber(age)) {ESP_LOGI(TAG, "Age: %d", age->valueint);}if (cJSON_IsBool(is_student)) {ESP_LOGI(TAG, "Is Student: %s", cJSON_IsTrue(is_student) ? "true" : "false");}// 删除解析后的 JSON 对象cJSON_Delete(root);
}void app_main(void) {// 创建 JSON 数据char *json_string = create_json_data();if (json_string != NULL) {ESP_LOGI(TAG, "Created JSON data: %s", json_string);// 解析 JSON 数据parse_json_data(json_string);// 释放 JSON 字符串内存cJSON_free(json_string);}
}

代码解释

  1. 引入头文件:引入了 stdio.hstring.hesp_log.h 和 cJSON.h 头文件,分别用于标准输入输出、字符串处理、ESP - IDF 日志和 JSON 处理。
  2. 创建 JSON 数据create_json_data() 函数创建一个新的 JSON 对象,并添加键值对。然后将 JSON 对象转换为字符串并返回。
  3. 解析 JSON 数据parse_json_data() 函数解析传入的 JSON 字符串,并读取其中的键值对。最后打印出读取到的值。
  4. 主函数:在 app_main() 函数中,调用 create_json_data() 函数创建 JSON 数据,打印创建的 JSON 字符串,然后调用 parse_json_data() 函数解析 JSON 数据。最后释放 JSON 字符串占用的内存。

编译和运行

将上述代码保存为一个 .c 文件,例如 main.c,并将其放置在 ESP - IDF 项目的 main 目录下。然后在终端中进入项目目录,执行以下命令进行编译和烧录:

idf.py build
idf.py -p /dev/ttyUSB0 flash monitor

请将 /dev/ttyUSB0 替换为你的 ESP32 - S3 开发板所使用的串口设备。

注意事项

  • 本示例使用了 cJSON 库,该库是一个轻量级的 JSON 解析库,已经集成在 ESP - IDF 中。
  • 在使用 cJSON 库时,需要注意内存管理,避免内存泄漏。例如,在创建 JSON 对象和打印 JSON 字符串后,需要调用 cJSON_Delete() 和 cJSON_free() 函数释放内存。

=================================================================

以下是几个在 C 语言中多行书写包含字符串和整型混合值的 JSON 键值对数据的例子,分别展示了不同的实现方式。

方式一:利用相邻字符串常量自动拼接特性

#include <stdio.h>int main() {const char *json_data = "{""\"name\": \"John Doe\",""\"age\": 30,""\"city\": \"New York\",""\"id\": 12345,""\"email\": \"johndoe@example.com\"""}";printf("%s\n", json_data);return 0;
}
代码解释:
  • 编译器会自动把相邻的字符串常量拼接成一个完整的字符串。这里将 JSON 的各个键值对拆分成多行书写,其中 "name" 和 "city""email" 的值是字符串,而 "age" 和 "id" 的值是整型。
  • 最后使用 printf 函数将拼接好的 JSON 字符串输出。

方式二:使用字符串拼接函数

#include <stdio.h>
#include <string.h>int main() {char json[512] = "{ ";// 拼接字符串值的键值对strcat(json, "\"name\": \"John Doe\", ");// 拼接整型值的键值对char age_str[10];sprintf(age_str, "\"age\": %d, ", 30);strcat(json, age_str);strcat(json, "\"city\": \"New York\", ");char id_str[10];sprintf(id_str, "\"id\": %d, ", 12345);strcat(json, id_str);strcat(json, "\"email\": \"johndoe@example.com\" ");strcat(json, "}");printf("%s\n", json);return 0;
}
代码解释:
  • 首先定义一个足够大的字符数组 json 用于存储拼接后的 JSON 字符串,并初始化为 作为 JSON 对象的起始。
  • 对于字符串值的键值对,直接使用 strcat 函数进行拼接。
  • 对于整型值的键值对,先使用 sprintf 函数将整型值转换为字符串格式,再将其拼接到 json 数组中。
  • 最后拼接上结束的花括号 },并使用 printf 函数输出最终的 JSON 字符串。

方式三:使用宏定义辅助拼接

#include <stdio.h>#define JSON_KEY_VALUE(key, value) "\"" #key "\": " #value ","int main() {const char *json_data = "{"JSON_KEY_VALUE(name, "John Doe")JSON_KEY_VALUE(age, 30)JSON_KEY_VALUE(city, "New York")JSON_KEY_VALUE(id, 12345)"\"email\": \"johndoe@example.com\"""}";printf("%s\n", json_data);return 0;
}
代码解释:
  • 定义了一个宏 JSON_KEY_VALUE,它可以将键和值转换为 JSON 键值对的字符串格式。
  • 在构建 JSON 字符串时,使用这个宏来处理大部分的键值对,对于一些特殊情况(如最后一个键值对不需要逗号)则直接书写。
  • 最后将拼接好的 JSON 字符串输出。

这几种方式都能实现多行书写包含字符串和整型混合值的 JSON 键值对数据,你可以根据实际情况选择最合适的方法。

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

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

相关文章

MyBatis-Plus 使用 Wrapper 构建动态 SQL 有哪些优劣势?

MyBatis-Plus (MP) 提供的 Wrapper (如 QueryWrapper, LambdaQueryWrapper, UpdateWrapper, LambdaUpdateWrapper) 是其核心特性之一&#xff0c;它允许我们在开发时以面向对象的方式构建 SQL 的 WHERE 条件、ORDER BY、SELECT 字段列表等部分。与传统的 MyBatis 在 XML 文件中…

STM32与i.MX6ULL内存与存储机制全解析:从微控制器到应用处理器的设计差异

最近做FreeRTos&#xff0c;以及前面设计的RVOS&#xff0c;这种RTOS级别的系统内存上的分布与CortexA系列里面的分布有相当大的区别&#xff0c;给我搞糊涂了。 目录 STM32&#xff08;Cortex-M系列&#xff09;的内存与存储机制 Flash存储内容RAM存储内容启动与运行时流程示例…

Eteam 0.3版本开发规划

Eteam 0.1系列经历了3个小版本&#xff0c;主要完成了团队资料库功能。 Eteam 0.2系列经历了22个小版本&#xff0c;主要完成了白板和AI交互的能力。 目前的问题 目前白板上的数据有两个来源&#xff0c;团队资料库和外部数据。外部数据和团队资料库数据边界不是很清晰。 0.3版…

HTML5好看的水果蔬菜在线商城网站源码系列模板5

文章目录 1.设计来源1.1 主界面1.2 关于我们1.3 商品服务1.4 果蔬展示1.5 联系我们1.6 商品具体信息1.7 登录注册 2.效果和源码2.1 动态效果2.2 源代码 源码下载万套模板&#xff0c;程序开发&#xff0c;在线开发&#xff0c;在线沟通 作者&#xff1a;xcLeigh 文章地址&#…

深入理解Java包装类:自动装箱拆箱与缓存池机制

深入理解Java包装类&#xff1a;自动装箱拆箱与缓存池机制 对象包装器 Java中的数据类型可以分为两类&#xff1a;基本类型和引用类型。作为一门面向对象编程语言&#xff0c; 一切皆对象是Java语言的设计理念之一。但基本类型不是对象&#xff0c;无法直接参与面向对象操作&…

uniapp自定义拖拽排列

uniapp自定义拖拽排列并改变下标 <!-- 页面模板 --> <template><view class"container"><view v-for"(item, index) in list" :key"item.id" class"drag-item" :style"{transform: translate(${activeInde…

基于SpringBoot的课程管理系统

前言 今天给大家分享一个基于SpringBoot的课程管理系统。 1 系统介绍 课程管理系统是一种专门为学校设计的软件系统&#xff0c;旨在帮助学校高效地管理和组织各类课程信息。 该系统通常包括学生、教师和管理员三大角色。 他们可以通过系统进行选课、查看课程表、考试、进…

max31865典型电路

PT100读取有很多种方案&#xff0c;常用的惠斯通电桥&#xff0c;和专用IC max31865 。 电阻温度检测器(RTD)是一种阻值随温度变化的电阻。铂是最常见、精度最高的测温金属丝材料。铂RTD称为PT-RTD&#xff0c;镍、铜和其它金属亦可用来制造RTD。RTD具有较宽的测温范围&#x…

飞算 JavaAI 与 Spring Boot:如何实现微服务开发效率翻倍?

微服务架构凭借其高内聚、低耦合的特性&#xff0c;成为企业构建复杂应用系统的首选方案。然而&#xff0c;传统微服务开发流程中&#xff0c;从服务拆分、接口设计到代码编写、调试部署&#xff0c;往往需要耗费大量时间与人力成本。Spring Boot 作为 Java 领域最受欢迎的微服…

(2)Docker 常用命令

文章目录 Docker 服务器Docker 镜像Docker 容器本地 RegistryRUN vs CMD vs ENTRYPOINTRUNCMDENTRYPOINT 限制容器对内存、CPU 和 IO 资源的使用内存CPUBlock IO设置权重bps 和 iops cgroup 和 namespacecgroupnamespacMount namespaceUTS namespaceIPC namespacePID namespace…

Django 实现电影推荐系统:从搭建到功能完善(附源码)

前言&#xff1a;本文将详细介绍如何使用 Django 构建一个电影推荐系统&#xff0c;涵盖项目的搭建、数据库设计、视图函数编写、模板渲染以及用户认证等多个方面。&#x1f517;软件安装、环境准备 ❤ 【作者主页—&#x1f4da;阅读更多优质文章、获取更多优质源码】 目录 一…

C#进阶学习(七)常见的泛型数据结构类(2)HashSet和SortedSet

目录 using System.Collections.Generic; // 核心命名空间 一、 HashSet 核心特性 常用方法 属性 二、SortedSet 核心特性 1、整型&#xff08;int、long 等&#xff09; 2、字符串型&#xff08;string&#xff09; 3、字符型&#xff08;char&#xff09; 4、自定义…

SQL之DML(查询语句:select、where)

&#x1f3af; 本文专栏&#xff1a;MySQL深入浅出 &#x1f680; 作者主页&#xff1a;小度爱学习 select查询语句 在开发中&#xff0c;查询语句是使用最多&#xff0c;也是CRUD中&#xff0c;复杂度最高的sql语句。 查询的语法结构 select *|字段1 [, 字段2 ……] from 表…

vue | 不同 vue 版本对复杂泛型的支持情况 · vue3.2 VS vue3.5

省流总结&#xff1a;defineProps 的泛型能力&#xff0c;来直接推导第三方组件的 props 类型 引入第三方库的类型&#xff0c;并直接在 <script setup> 中作为 props 使用。这种类型一般是复杂泛型&#xff08;包含联合类型、可选属性、交叉类型、条件类型等&#xff0…

Unity-无限滚动列表实现Timer时间管理实现

今天我们来做一个UI里经常做的东西&#xff1a;无限滚动列表。 首先我们得写清楚实现的基本思路&#xff1a; 所谓的无限滚动当然不是真的无限滚动&#xff0c;我们只要把离开列表的框再丢到列表的后面就行&#xff0c;核心理念和对象池是类似的。 我们来一点一点实现&#x…

Docker的基本概念和一些运用场景

Docker 是一种开源的容器化平台&#xff0c;可以帮助开发人员更加高效地打包、发布和运行应用程序。以下是 Docker 的基本概念和优势&#xff1a; 基本概念&#xff1a; 容器&#xff1a;Docker 使用容器来打包应用程序及其依赖项&#xff0c;容器是一个独立且可移植的运行环境…

Unity中基于第三方插件扩展的对于文件流处理的工具脚本

在Unity的项目中对应文件处理,在很多地方用到,常见的功能,就是保存文件,加载文件,判断文件或者文件夹是否存在,删除文件等。 在之前已经写过通过C#的IO实现的这些功能,可查看《Unity C# 使用IO流对文件的常用操作》,但是不能保证所有平台都可以使用 现在基于第三方跨…

Flink介绍——实时计算核心论文之MillWheel论文详解

引入 通过前面的文章&#xff0c;我们从S4到Storm&#xff0c;再到Storm结合Kafka成为当时的实时处理最佳实践&#xff1a; S4论文详解S4论文总结Storm论文详解Storm论文总结Kafka论文详解Kafka论文总结 然而KafkaStorm的第一代流式数据处理组合&#xff0c;还面临的三个核心…

python异步协程async调用过程图解

1.背景&#xff1a; 项目中有用到协程&#xff0c;但是对于协程&#xff0c;线程&#xff0c;进程的区别还不是特别了解&#xff0c;所以用图示的方式画了出来&#xff0c;用于理清三者的概念。 2.概念理解&#xff1a; 2.1协程&#xff0c;线程&#xff0c;进程包含关系 一…

【React】获取元素距离页面顶部的距离

文章目录 代码实现 代码实现 import { useEffect, useRef, useState } from react;const DynamicPositionTracker () > {const [distance, setDistance] useState(0);const divRef useRef(null);useEffect(() > {const targetDiv divRef.current;if (!targetDiv) re…