如何用Python处理JSON嵌套数据?json_normalize技巧

json_normalize处理多层嵌套json的关键在于record_path和meta参数的配合使用。1. record_path用于指定要展开的列表路径,可以是字符串或列表形式,如’orders’或[‘orders’, ‘items’],表示逐层展开;2. meta用于保留父级字段信息,可指定单层或多层路径,如[‘contact’, ’email’];3. 处理不规则结构时,可通过errors=’ignore’忽略缺失键,用nan填充;4. 拍平后的dataframe可结合pandas进行数据类型转换、列重命名、缺失值处理、数据聚合等操作,实现高效分析与清洗。

如何用Python处理JSON嵌套数据?json_normalize技巧

Python处理JSON嵌套数据,json_normalize是一个非常实用的工具,它能将复杂的嵌套结构“拍平”成表格形式,便于数据分析和处理,尤其在处理API返回或日志数据时,效率极高。

如何用Python处理JSON嵌套数据?json_normalize技巧

解决方案

在数据处理的日常中,我们经常会遇到JSON数据,尤其是那些层层嵌套、结构复杂的JSON。直接用字典方式去一层层取值,不仅代码臃肿,还容易出错,特别是当某个键可能不存在时。这时候,json_normalize就显得非常方便了,它能把这种树状结构转换成扁平的Pandas DataFrame,就像把一张立体地图摊平了看。

我们来看一个典型的场景:一份包含用户、订单及其商品详情的JSON数据。

立即学习“Python免费学习笔记(深入)”;

如何用Python处理JSON嵌套数据?json_normalize技巧

[  {    "user_id": "U001",    "user_name": "Alice",    "contact": {      "email": "alice@example.com",      "phone": "123-456-7890"    },    "orders": [      {        "order_id": "O101",        "date": "2023-01-15",        "items": [          {"item_id": "P001", "name": "Laptop", "price": 1200, "qty": 1},          {"item_id": "P002", "name": "Mouse", "price": 25, "qty": 2}        ]      },      {        "order_id": "O102",        "date": "2023-01-20",        "items": [          {"item_id": "P003", "name": "Keyboard", "price": 75, "qty": 1}        ]      }    ]  },  {    "user_id": "U002",    "user_name": "Bob",    "contact": {      "email": "bob@example.com",      "phone": "987-654-3210"    },    "orders": [      {        "order_id": "O201",        "date": "2023-02-01",        "items": [          {"item_id": "P004", "name": "Monitor", "price": 300, "qty": 1}        ]      }    ]  }]

要将上述数据“拍平”,尤其是提取ordersitems中的信息,同时保留用户和订单的基本信息,json_normalizerecord_pathmeta参数就派上用场了。

import pandas as pdfrom pandas import json_normalizeimport jsondata = [  {    "user_id": "U001",    "user_name": "Alice",    "contact": {      "email": "alice@example.com",      "phone": "123-456-7890"    },    "orders": [      {        "order_id": "O101",        "date": "2023-01-15",        "items": [          {"item_id": "P001", "name": "Laptop", "price": 1200, "qty": 1},          {"item_id": "P002", "name": "Mouse", "price": 25, "qty": 2}        ]      },      {        "order_id": "O102",        "date": "2023-01-20",        "items": [          {"item_id": "P003", "name": "Keyboard", "price": 75, "qty": 1}        ]      }    ]  },  {    "user_id": "U002",    "user_name": "Bob",    "contact": {      "email": "bob@example.com",      "phone": "987-654-3210"    },    "orders": [      {        "order_id": "O201",        "date": "2023-02-01",        "items": [          {"item_id": "P004", "name": "Monitor", "price": 300, "qty": 1}        ]      }    ]  }]# 第一次拍平:将用户和订单关联起来# record_path 指定要展开的列表路径# meta 指定要保留的父级键orders_df = json_normalize(    data,    record_path='orders',    meta=['user_id', 'user_name', ['contact', 'email'], ['contact', 'phone']],    sep='_' # 用于连接meta中多层嵌套键的名称)# 第二次拍平:将订单和商品关联起来# 此时的输入数据是orders_df,但我们需要操作的是其中的'items'列# 这一步稍微有点技巧,因为json_normalize通常直接处理list of dicts# 这里需要对orders_df的每一行进行迭代或再次应用json_normalize# 更直接的方法是先将所有items提取出来,再normalizeall_items = []for idx, row in orders_df.iterrows():    order_items = row['items']    # 将订单ID和日期等信息添加到每个item中,以便后续关联    for item in order_items:        item['order_id'] = row['order_id']        item['order_date'] = row['date']        item['user_id'] = row['user_id'] # 再次添加用户ID,方便最终合并    all_items.extend(order_items)items_df = json_normalize(all_items)# 最后,将用户、订单、商品信息合并成一个宽表# 通常我们会选择一个合适的键进行合并,这里是user_id和order_id# 但由于第二次拍平已经包含了这些信息,我们只需要选择需要的列# 也可以考虑先将所有数据normalize到最细粒度(item),再选择列final_df = items_df[[    'user_id', 'order_id', 'order_date',    'item_id', 'name', 'price', 'qty']]print(final_df)

这段代码会输出一个扁平化的DataFrame,每一行代表一个商品,并附带了其所属订单和用户的相关信息。这种分步处理的方式,在面对多层嵌套时,能让逻辑更清晰。

如何用Python处理JSON嵌套数据?json_normalize技巧

处理复杂多层嵌套JSON时,json_normalize的record_path和meta参数怎么用?

json_normalize的核心魅力,确实在于record_pathmeta这两个参数的灵活运用。它们像是两把钥匙,一把用来打开你要展开的“列表之门”,另一把则帮你把“门外”的上下文信息带进来。

record_path参数是用来指定JSON中哪个列表(或列表中的字典)应该被展开成新的行。它可以是一个字符串,比如'orders',表示直接展开顶层下的orders列表。如果嵌套更深,比如要展开orders列表中的每个订单里的items列表,那么record_path就应该是一个路径列表,例如['orders', 'items']。这表示json_normalize会先进入orders,然后对orders里的每个元素(也就是每个订单字典)再进入items列表进行展开。

举个例子,如果我们想直接从原始数据中获取所有商品的信息,同时保留其所属的用户ID和订单ID,record_pathmeta的组合就会是这样:

# 假设我们想直接从最顶层的数据中,一步到位地获取所有商品的详细信息,# 并关联上用户ID、用户姓名、订单ID和订单日期。# 这就需要record_path指向['orders', 'items']# meta则需要包含 user_id, user_name, 以及 orders下的 order_id 和 date# 注意:当record_path指向一个多层路径时,meta中的路径也需要相应调整# 比如,如果record_path是 ['orders', 'items'],那么meta中的 'order_id'# 实际上是从 'orders' 这一层级获取的,所以需要写成 ['orders', 'order_id']# 但json_normalize在处理这种多层record_path时,meta参数的解读会有些不同# 它会默认你提供的meta字段是相对于record_path的“父级”层级。# 简单来说,如果record_path是A->B,那么meta中的字段就是从A这个层级取的。# 但如果meta字段本身也是嵌套的,比如 contact.email,就需要用列表表示 ['contact', 'email']# 实际操作中,直接用['orders', 'items']作为record_path,并把所有父级信息都放进meta,# 可能会有点复杂,因为meta的路径是相对于record_path的父级而言的。# 通常更推荐分步处理,或者先将数据结构预处理一下。# 但如果结构允许,可以这样尝试:items_flat_df = json_normalize(    data,    record_path=['orders', 'items'], # 展开到最细粒度的items    meta=[        'user_id',        'user_name',        ['contact', 'email'],        ['contact', 'phone'],        ['orders', 'order_id'], # 从orders层级获取order_id        ['orders', 'date']      # 从orders层级获取date    ],    errors='ignore' # 忽略可能存在的路径错误,避免中断)# 这样处理后,你会发现['orders', 'order_id']和['orders', 'date']可能会出现重复,# 因为json_normalize会为每个item重复其父级orders的信息。# 实际输出时,它会尝试将这些父级信息关联到每个展开的子项上。# 但需要注意的是,当record_path是多层嵌套时,meta的路径是相对于record_path的*直接父级*而言的。# 也就是说,如果record_path是 ['orders', 'items'],那么 meta 里的 'order_id' # 是从 'orders' 这个层级取出来的,而不是从最顶层。# 这也是为什么我个人倾向于分步拍平,或者在第二次拍平前,先将第一次拍平的结果进行预处理,# 把需要保留的父级信息直接注入到子级列表的每个字典中。# 比如,在上面第一次拍平后,迭代`orders_df`,把`order_id`和`user_id`加到每个`item`字典里,# 这样第二次拍平`items`时,它们就自然成为列了。这种手动注入的方式,虽然多了一步,# 但在处理逻辑上会更直观,尤其在JSON结构非常复杂且不规则时。`meta`参数则负责从原始JSON的父级层中提取你想要保留的字段。它可以是一个字符串(如`'user_id'`),也可以是一个路径列表(如`['contact', 'email']`),用于提取嵌套的父级字段。`json_normalize`会把这些`meta`字段的值复制到每个展开的行中,确保你不会丢失上下文信息。当你需要将不同层级的数据关联起来时,`meta`是必不可少的。### json_normalize遇到缺失值或不规则结构时如何应对?实际工作中,我们拿到的JSON数据很少是完美的。键可能缺失,或者某些字段的结构突然变了,这在日志数据或者第三方API返回中尤其常见。`json_normalize`在处理这些“不完美”时,默认行为是相当稳健的。当`json_normalize`在尝试访问`record_path`或`meta`中指定的键,而该键不存在时,它会默认用`NaN`(Not a Number)或`None`来填充对应的列。这通常是可接受的,因为它避免了程序崩溃,并清晰地标识了数据缺失的位置。如果你希望在遇到这种缺失或不规则情况时,`json_normalize`的行为有所不同,可以使用`errors`参数。*   `errors='ignore'` (默认值): 这是最常用的选项,它会忽略错误,将无法解析的字段填充为`NaN`,然后继续处理。这对于数据质量不高的场景非常有用,因为它能让你尽可能多地提取出有效数据。*   `errors='raise'`: 如果你对数据结构有严格要求,任何一个指定路径的键缺失都应该被视为错误并中断程序,那么可以选择这个选项。它会抛出一个`KeyError`或其他相关异常。这在开发和测试阶段,或者你确信数据应该总是符合特定结构时很有用,能帮助你快速发现数据源的问题。举个例子,如果我们的JSON数据中,有的用户没有`contact`信息,或者`orders`列表是空的:```json[  {    "user_id": "U003",    "user_name": "Charlie",    "orders": [] # 空订单列表  },  {    "user_id": "U004",    "user_name": "Diana",    "contact": { "email": "diana@example.com" } # 缺少phone  }]

当我们用之前的json_normalize方法处理时:

data_irregular = [  {    "user_id": "U003",    "user_name": "Charlie",    "orders": []  },  {    "user_id": "U004",    "user_name": "Diana",    "contact": { "email": "diana@example.com" }  }]# 第一次拍平orders_df_irregular = json_normalize(    data_irregular,    record_path='orders',    meta=['user_id', 'user_name', ['contact', 'email'], ['contact', 'phone']],    sep='_',    errors='ignore' # 明确指定忽略错误)print(orders_df_irregular)

对于U003,因为orders列表为空,json_normalize不会生成任何行,这很合理。对于U004contact_phone列会显示NaN,因为它在原始JSON中不存在。这种处理方式,让我觉得json_normalize在鲁棒性方面做得相当不错,能应对很多实际场景中的数据脏乱问题。当然,如果结构不规则到连record_path本身都可能不是列表,或者meta路径下的值类型不一致,那可能就需要一些预处理,比如用try-except块手动解析,或者在json_normalize之前用列表推导式清理数据。

json_normalize处理后的数据,如何结合Pandas进行高效分析和清洗?

json_normalize的输出是一个Pandas DataFrame,这意味着我们可以无缝地利用Pandas的强大功能进行后续的数据分析和清洗。这才是真正发挥数据价值的关键一步。

一旦你把复杂的JSON结构“拍平”了,接下来通常会做这些事情:

数据类型转换: json_normalize默认会尝试推断列的数据类型,但有时推断不准确,比如数字被当作字符串,日期被当作对象。这时候,可以使用df['column_name'].astype(int)pd.to_datetime(df['date_column'])等方法进行显式转换。例如,商品价格和数量应该转换为数值类型,订单日期应该转换为日期时间类型,这对于后续的数值计算和时间序列分析至关重要。

# 假设items_df已经生成items_df['price'] = pd.to_numeric(items_df['price'], errors='coerce') # errors='coerce' 将无法转换的值设为NaNitems_df['qty'] = pd.to_numeric(items_df['qty'], errors='coerce')items_df['order_date'] = pd.to_datetime(items_df['order_date'])

列重命名与选择: json_normalize在处理嵌套键时,会默认用下划线连接,比如contact_email。如果你觉得这些列名不够直观,或者想简化,可以利用df.rename(columns={'old_name': 'new_name'})进行批量重命名。同时,如果拍平后出现了大量你不需要的中间列,可以使用df[['col1', 'col2', ...]]进行列选择,只保留你关心的核心数据。

# 假设我们想把contact_email改成更简洁的emailfinal_df_renamed = final_df.rename(columns={'contact_email': 'email'})# 或者直接在选择列的时候就完成简化# final_df = items_df[['user_id', 'order_id', 'order_date', 'item_id', 'name', 'price', 'qty', 'contact_email']]# final_df = final_df.rename(columns={'contact_email': 'email'})

数据清洗与缺失值处理: 拍平后的DataFrame可能会有NaN值,这需要根据业务逻辑进行处理。你可以选择填充(df.fillna(value))、删除(df.dropna())或者进行更复杂的插值。

# 填充缺失的电话号码为'未知'final_df['contact_phone'].fillna('未知', inplace=True)

数据聚合与透视: 这是数据分析的核心。一旦数据扁平化,你就可以轻松地进行分组聚合(df.groupby()),比如计算每个用户的总消费、每个商品的销售总量等。也可以使用pivot_table进行数据透视,从不同维度观察数据。

# 计算每个用户的总消费user_total_spend = final_df.groupby('user_id')['price'].sum()print("用户总消费:n", user_total_spend)# 计算每个商品的销售总量item_sales_qty = final_df.groupby('item_id')['qty'].sum()print("n商品销售总量:n", item_sales_qty)# 查看每个订单的商品数量order_item_counts = final_df.groupby('order_id')['item_id'].count()print("n订单商品数量:n", order_item_counts)

与其他数据源合并: 在实际项目中,你可能需要将这份拍平的JSON数据与来自数据库、CSV文件等其他数据源的数据进行合并(pd.merge()),以构建更全面的分析视图。例如,将用户ID与一个包含用户地理位置信息的DataFrame合并。

这些后续步骤,才是真正让json_normalize处理后的数据发挥其潜力的环节。它不仅仅是一个工具,更是一个数据处理流程中的关键“中转站”,将复杂的数据结构转化为易于操作的表格形式,为后续的深度分析铺平道路。

以上就是如何用Python处理JSON嵌套数据?json_normalize技巧的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1363783.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
Pandas cut 函数进阶:理解输出与定制分箱区间
上一篇 2025年12月14日 03:36:57
Python:无需 itertools 库,垂直打印多字符串
下一篇 2025年12月14日 03:37:15

相关推荐

  • composer require-dev和require有什么不同_Composer Require与Require-Dev区别解析

    require用于声明项目运行必需的依赖,如框架、数据库组件和第三方SDK,这些包会随项目部署到生产环境;2. require-dev用于声明仅在开发和测试阶段需要的工具,如PHPUnit、PHPStan、Faker等,不会默认部署到生产环境;3. 安装时composer install根据环境决定…

    2026年5月10日
    1000
  • Matplotlib 地图中多类型图例的创建与优化

    Matplotlib 地图中多类型图例的创建与优化Matplotlib 地图中多类型图例的创建与优化Matplotlib 地图中多类型图例的创建与优化Matplotlib 地图中多类型图例的创建与优化

    本教程旨在解决matplotlib地图可视化中,如何在一个图例中同时展示颜色块(如区域分类)和自定义标记(如特定兴趣点)的问题。文章详细介绍了当传统`patch`对象无法正确显示标记时,如何利用`matplotlib.lines.line2d`创建标记图例句柄,并将其与颜色块图例句柄合并,从而生成一…

    2026年5月10日 用户投稿
    100
  • Golang JSON序列化:控制敏感字段暴露的最佳实践

    本教程探讨golang中如何高效控制结构体字段在json序列化时的可见性。当需要将包含敏感信息的结构体数组转换为json响应时,通过利用`encoding/json`包提供的结构体标签,特别是`json:”-“`,可以轻松实现对特定字段的忽略,从而避免敏感数据泄露,确保api…

    2026年5月10日
    000
  • 利用海象运算符简化条件赋值:Python教程与最佳实践

    本文旨在探讨Python中海象运算符(:=)在条件赋值场景下的应用。通过对比传统if/else语句与海象运算符,以及条件表达式,分析海象运算符在简化代码、提高可读性方面的优势与局限性。并通过具体示例,展示如何在列表推导式等场景下合理使用海象运算符,同时强调其潜在的复杂性及替代方案,帮助开发者更好地掌…

    2026年5月10日
    100
  • Debian syslog性能优化技巧有哪些

    提升Debian系统syslog (通常基于rsyslog)性能,关键在于精简配置和高效处理日志。以下策略能有效优化日志管理,提升系统整体性能: 精简配置,高效加载: 在rsyslog配置文件中,仅加载必要的输入、输出和解析模块。 使用全局指令设置日志级别和格式,避免不必要的处理。 自定义模板: 创…

    2026年5月10日
    000
  • 比特币新手教程 比特币交易平台有哪些

    比特币是一种去中心化的数字货币,基于区块链技术实现点对点交易,具有匿名性、有限发行和不可篡改等特点;新手可通过交易所购买,P2P交易获得比特币,常用平台包括Binance、OKX和Huobi;交易流程包括注册账户、实名认证、绑定支付方式、充值法币并下单购买,可选择市价单或限价单;比特币存储方式有交易…

    2026年5月10日
    000
  • c++中的SFINAE技术是什么_c++模板编程中的SFINAE原理与应用

    SFINAE 是“替换失败不是错误”的原则,指模板实例化时若参数替换导致错误,只要存在其他合法候选,编译器不报错而是继续重载决议。它用于条件启用模板、类型检测等场景,如通过 decltype 或 enable_if 控制函数重载,实现类型特征判断。尽管 C++20 引入 Concepts 简化了部分…

    2026年5月10日
    000
  • Go语言mgo查询构建:深入理解bson.M与日期范围查询的正确实践

    本文旨在解决go语言mgo库中构建复杂查询时,特别是涉及嵌套`bson.m`和日期范围筛选的常见错误。我们将深入剖析`bson.m`的类型特性,解释为何直接索引`interface{}`会导致“invalid operation”错误,并提供一种推荐的、结构清晰的代码重构方案,以确保查询条件能够正确…

    2026年5月10日
    100
  • RichHandler与Rich Progress集成:解决显示冲突的教程

    在使用rich库的`richhandler`进行日志输出并同时使用`progress`组件时,可能会遇到显示错乱或溢出问题。这通常是由于为`richhandler`和`progress`分别创建了独立的`console`实例导致的。解决方案是确保日志处理器和进度条组件共享同一个`console`实例…

    2026年5月10日
    000
  • 理解编程指令:当结果正确,但实现方式不符要求时

    本文探讨了在编程实践中,即使程序输出了正确的结果,但若其实现方式未能严格遵循既定指令,仍可能被视为“不正确”的问题。我们将通过具体示例,对比直接求和与累加求和两种实现策略,强调理解和遵守编程规范的重要性,以确保代码的健壮性、可维护性及符合项目要求。 在软件开发过程中,我们经常会遇到这样的情况:编写的…

    2026年5月10日
    000
  • Golang goroutine与channel调试技巧

    使用go run -race检测数据竞争,结合runtime.NumGoroutine监控协程数量,通过pprof分析阻塞调用栈,利用select超时避免永久阻塞,有效排查goroutine泄漏、死锁和数据竞争问题。 Go语言的goroutine和channel是并发编程的核心,但它们也带来了调试上…

    2026年5月10日
    000
  • 《魔兽世界》将于6月11日开启国服回归技术测试

    《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试

    《%ign%ignore_a_1%re_a_1%》官方宣布,将于6月11日开启国服回归技术测试,时间为7天,并称可以在6月内正式开服,玩家们可以访问官网下载战网客户端并预下载“巫妖王之怒”客户端,技术测试详情见下图。 WordAi WordAI是一个AI驱动的内容重写平台 53 查看详情 以上就是《…

    2026年5月10日 用户投稿
    200
  • 使用 Jupyter Notebook 进行探索性数据分析

    Jupyter Notebook通过单元格实现代码与Markdown结合,支持数据导入(pandas)、清洗(fillna)、探索(matplotlib/seaborn可视化)、统计分析(describe/corr)和特征工程,便于记录与分享分析过程。 Jupyter Notebook 是进行探索性…

    2026年5月10日
    000
  • 如何在HTML中插入表单元素_HTML表单控件与输入类型使用指南

    HTML表单通过标签构建,包含action和method属性定义数据提交目标与方式,常用input类型如text、password、email等适配不同输入需求,配合label、required、placeholder提升可用性,结合textarea、select、button等控件实现完整交互,是…

    2026年5月10日
    000
  • 网站标题关键词更新后,搜索引擎为何仍显示旧标题?

    网站标题更新后,搜索引擎为何显示旧标题? 网站SEO优化中,站长常修改网站标题关键词,期望搜索结果显示自定义标题。然而,即使更新标签、meta keywords、meta description和结构化数据中的name属性后,搜索结果仍显示旧标题,这令人费解。本文将对此进行解释。 问题:站长修改了网…

    2026年5月10日
    100
  • 创建指定大小并填充特定数据的Golang文件教程

    本文将介绍如何使用Golang创建一个指定大小的文件,并用特定数据填充它。我们将使用 `os` 包提供的函数来创建和截断文件,从而实现快速生成大文件的目的。示例代码展示了如何创建一个10MB的文件,并将其填充为全零数据。掌握这些方法,可以方便地在例如日志系统或磁盘队列等场景中,预先创建测试文件或初始…

    2026年5月10日
    000
  • Python命令怎样使用profile分析脚本性能 Python命令性能分析的基础教程

    使用Python的cProfile模块分析脚本性能最直接的方式是通过命令行执行python -m cProfile your_script.py,它会输出每个函数的调用次数、总耗时、累积耗时等关键指标,帮助定位性能瓶颈;为进一步分析,可将结果保存为文件python -m cProfile -o ou…

    2026年5月10日
    000
  • 如何插入查询结果数据_SQL插入Select查询结果方法

    如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法

    使用INSERT INTO…SELECT语句可高效插入数据,通过NOT EXISTS、LEFT JOIN、MERGE语句或唯一约束避免重复;表结构不一致时可通过别名、类型转换、默认值或计算字段处理;结合存储过程可提升可维护性,支持参数化与动态SQL。 将查询结果数据插入到另一个表中,可以…

    2026年5月10日 用户投稿
    000
  • 使用 WebCodecs VideoDecoder 实现精确逐帧回退

    本文档旨在解决在使用 WebCodecs VideoDecoder 进行视频解码时,实现精确逐帧回退的问题。通过比较帧的时间戳与目标帧的时间戳,可以避免渲染中间帧,从而提高用户体验。本文将提供详细的解决方案和示例代码,帮助开发者实现精确的视频帧控制。 在使用 WebCodecs VideoDecod…

    2026年5月10日
    000
  • Discord.py 交互按钮超时与持久化解决方案

    本教程旨在解决Discord.py中交互按钮在一段时间后出现“This Interaction Failed”错误的问题。我们将深入探讨视图(View)的超时机制,并提供通过正确设置timeout参数以及利用bot.add_view()方法实现按钮持久化的具体方案,确保您的机器人交互功能稳定可靠,即…

    2026年5月10日
    000

发表回复

登录后才能评论
关注微信