【常见开源库的二次开发】一文学懂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://xiahunao.cn/news/3226362.html

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

CentOS7系统上安装MySQL8.0(rpm-bundle.tar)详细过程

一、MySQL官网下载安装包 1.进入官网MySQL :: Download MySQL Community Server 2.查看自己的版本和架构 uname -mcat /etc/redhat-release 3.选择对应版本并下载 4.查看linux自带的mariadb数据库&#xff0c;有就卸载掉。 rpm -qa | grep mariadbrpm -e mariadb-libs…

【卡尔曼滤波】高斯白噪声

生成高斯白噪声并将其应用于信号处理 生成高斯白噪声并将其应用于信号处理 #以下是一个生成高斯白噪声并将其应用于信号处理的示例代码:import numpy as np import matplotlib.pyplot as plt import matplotlib.font_manager ## not work#notice matplotlibrc is a file, not…

学生选课管理系统(Java+MySQL)

技术栈 Java: 用于实现系统的核心业务逻辑。MySQL: 作为关系型数据库&#xff0c;用于存储系统中的数据。JDBC: 用于Java程序与MySQL数据库之间的连接和交互。Swing GUI: 用于创建图形用户界面&#xff0c;提升用户体验。 系统功能 我们的学生选课管理系统主要针对学生和管理…

突破传统:实现智慧校园实习单位变更

在智慧校园的实习管理系统设计中&#xff0c;充分考虑到了实习阶段学生可能遇到的实际需求&#xff0c;特别是实习单位变更这一灵活性要求&#xff0c;系统特设了一套完善的在线处理机制&#xff0c;旨在促进学生、学校与企业间的顺畅沟通与协调&#xff0c;确保实习过程的平稳…

Gmail邮件提醒通知如何设置?有哪些方法?

Gmail邮件提醒通知功能怎么样&#xff1f;通知邮件怎么有效发送&#xff1f; Gmail作为全球广泛使用的电子邮件服务&#xff0c;提供了多种邮件提醒通知功能&#xff0c;帮助用户不错过重要信息。AokSend将详细介绍如何设置Gmail邮件提醒通知&#xff0c;确保您不会错过任何重…

IT审计必看!对比旧版,CISA考试改版升级亮点和重点内容是什么?

官方通知&#xff0c;今年8月1日&#xff0c;CISA新版考纲正式上线&#xff0c;旧版在7月23日后就无法约考了。 艾威培训邀请了国内知名的IT审计CISA授课老师吴老师来为大家详细讲解CISA新版考纲的变化 目前第28th版教材只有英文版&#xff0c;中文版尚未发布。我们艾威经验丰…

【NOI-题解】1108 - 正整数N转换成一个二进制数1290 - 二进制转换十进制1386 - 小丽找半个回文数1405 - 小丽找潜在的素数?

文章目录 一、前言二、问题问题&#xff1a;1108 - 正整数N转换成一个二进制数问题&#xff1a;1290 - 二进制转换十进制问题&#xff1a;1386 - 小丽找半个回文数问题&#xff1a;1405 - 小丽找潜在的素数&#xff1f; 三、感谢 一、前言 本章节主要对进制转换的题目进行讲解…

【UNI-APP】阿里NLS一句话听写typescript模块

阿里提供的demo代码都是javascript&#xff0c;自己捏个轮子。参考着自己写了一个阿里巴巴一句话听写Nls的typescript模块。VUE3的组合式API形式 startClient&#xff1a;开始听写&#xff0c;注意下一步要尽快开启识别和传数据&#xff0c;否则6秒后会关闭 startRecognition…

javascript高级部分笔记

javascript高级部分 Function方法 与 函数式编程 call 语法&#xff1a;call([thisObj[,arg1[, arg2[, [,.argN]]]]]) 定义&#xff1a;调用一个对象的一个方法&#xff0c;以另一个对象替换当前对象。 说明&#xff1a;call 方法可以用来代替另一个对象调用一个方法。cal…

侯捷C++面向对象高级编程(下)-2-non-explicit one argument constructor

1.构造函数 构造函数: Fraction(int num, int den 1) 初始化分子和分母&#xff0c;允许指定分子 num 和可选的分母 den。默认情况下&#xff0c;分母为 1。 加法运算符重载: Fraction operator(const Fraction& f) 重载了加法运算符 。这使得两个 Fraction 对象可以通过 …

NodeJS校园快递智能互助平台-计算机毕业设计源码58554

摘 要 随着校园人口的增加和生活节奏的加快&#xff0c;校园快递成为一个重要的服务需求。然而&#xff0c;传统的校园快递方式存在一些问题&#xff0c;例如无法满足快速和高效的需求&#xff0c;易发生丢失或损坏的情况&#xff0c;同时也给快递人员和用户带来不便。因此&am…

成功登上主要中心化交易所 (CEX) 的终极指南:从准备到上市的全面策略

对于区块链项目的创始人而言&#xff0c;成功的代币发行是项目发展的关键一步。尤其是在主要中心化交易所 (CEX) 上上市代币&#xff0c;可以极大地提高项目的曝光度和流动性。然而&#xff0c;CEX 上市过程复杂且充满挑战&#xff0c;需要创始人提前做好充分准备。本文将详细介…

JavaSE语法 | 初识Java!!!

初识Java 一、Java开发环境二、初步认识Java的main方法2.1 main方法的实现2.2 运行Java程序 三、注释四、标识符五、关键字 一、Java开发环境 IDEA版本&#xff1a;IntelliJ IDEA Community Edition 2022.3.3 JDK17 Windows 11 二、初步认识Java的main方法 2.1 main方法的实…

comsol multiphysics在岩土工程中的应用

comsol教程推荐&#xff1a; comsol multiphysics在岩土工程中的应用 [comsol multiphysics在岩土工程中的应用](https://download.csdn.net/download/qq_36980284/89529402) 出版发行项: 北京:中国建筑工业出版社,2014 ISBN及定价: 978-7-112-16188-1 CNY42.00 载体形态项:…

【Python】一文向您详细介绍 argparse中 action=‘store_false’ 的作用

【Python】一文向您详细介绍 argparse中 action‘store_false’ 的作用 下滑即可查看博客内容 &#x1f308; 欢迎莅临我的个人主页 &#x1f448;这里是我静心耕耘深度学习领域、真诚分享知识与智慧的小天地&#xff01;&#x1f387; &#x1f393; 博主简介&#xff1a…

邮件推送服务的自动化流程设置与优化技巧?

邮件推送服务如何定制化&#xff1f;邮件推送的安全性如何保障&#xff1f; 无论是大型企业还是小型企业&#xff0c;通过精准的邮件推送服务&#xff0c;可以实现客户关系管理的有效增强&#xff0c;提升品牌认知度和销售转化率。AokSend将探讨如何通过自动化流程设置与优化技…

天翼云高级运维工程师202407回忆题库 最新出炉

备考天翼云高级运维工程师 必须备考天翼云 之前觉得外企牛批 然后民企&#xff0c;拔地而起&#xff0c;民企也不错&#xff0c;工资高&#xff0c;有钱途 现在看来看去&#xff0c;还是国企好&#xff0c;体制内的&#xff0c;有保障&#xff0c;树大根深 有必要备考下天…

LLM+本地知识库?简单又没那么简单

最近在研究如何将大语言模型结合本地知识库进行问答&#xff0c;虽然网上已经有很多教程&#xff0c;但大部分都是基于LangChain进行文本分割&#xff0c;然后调用模型向量化的API。这种方式的确很简单&#xff0c;但有这么几个前提&#xff1a; 大模型不使用ChatGPT的话&…

07-04 周四 关于vLLM(LLMs_inference)源码安装过程问题与解决

07-04 周四 关于LLMs_inference源码安装过程问题与解决 时间版本修改人描述2024年7月4日09:48:09V0.1宋全恒新建文档 简介 由于最近需要向vLLM上集成功能&#xff0c;因此&#xff0c;需要能够调试自己的仓库LLMs_Inference&#xff0c;该文档记录了源码编译的完整的过程。 参…

对比:9款最佳个人项目管理软件盘点

文章介绍了9款个人项目管理软件&#xff1a;PingCode、Worktile、Flowus、Todoist、Trello、Teambition、有道云笔记、Notion、Microsoft To Do。 在管理个人项目时&#xff0c;是否常感到信息零散、进度难以把控&#xff1f;选择合适的项目管理软件&#xff0c;可以有效解决这…