C++23 新利器:深入解析栈踪迹库 (P0881R7)
文章目录
- 为何需要标准化的栈踪迹?
- P0881R7 的核心组件与使用
- 基本用法示例
- 与异常处理的集成
- 优势与价值
- 潜在的考量
- 总结
对于 C++ 开发者而言,调试和错误诊断一直是开发周期中不可或缺但又充满挑战的一环。当程序崩溃或发生未预期行为时,获取清晰、准确的调用栈信息至关重要。在 C++23 标准之前,开发者通常需要依赖平台特定的 API 或第三方库来实现这一功能,这不仅增加了代码的复杂性,也降低了可移植性。
令人振奋的是,C++23 标准正式引入了 栈踪迹库 (<stacktrace>
),其提案编号为 P0881R7。这一新特性为 C++ 开发者提供了一个标准化的、可移植的方式来捕获和操作程序当前的调用栈信息。
为何需要标准化的栈踪迹?
在 P0881R7 出现之前,获取栈踪迹的方法五花八门:
- 特定于编译器的内置函数: 例如 GCC 的
__builtin_return_address
和__builtin_frame_address
。 - 操作系统特定的 API: 例如 Windows 的
CaptureStackBackTrace
或 POSIX 系统中的backtrace
和backtrace_symbols
。 - 第三方库: 例如 Boost.Stacktrace 或 backward-cpp。
这些方法各有优缺点,但共同的问题在于:
- 可移植性差: 代码需要在不同平台和编译器之间进行条件编译和适配。
- API 差异: 不同方法的接口和功能各不相同,学习成本和维护成本较高。
- 集成难度: 将这些非标准化的工具集成到现有项目中可能比较复杂。
C++23 栈踪迹库的出现,旨在解决这些痛点,提供一个统一、简洁且强大的解决方案。
P0881R7 的核心组件与使用
新的 <stacktrace>
头文件引入了几个关键的类和函数:
-
std::stacktrace_entry
: 表示调用栈中的单个帧(frame)。它通常包含以下信息(具体可用性取决于实现和编译选项):- 源文件名:
source_file()
- 源文件行号:
source_line()
- 函数名 (可能经过修饰):
description()
(通常包含函数签名) - 原生句柄 (实现定义):
native_handle()
- 源文件名:
-
std::basic_stacktrace<Allocator>
: 表示一个栈踪迹,即std::stacktrace_entry
的集合。它是一个模板类,允许用户自定义内存分配器。标准库也提供了别名std::stacktrace
,使用默认的分配器。- 获取当前栈踪迹:
static std::basic_stacktrace current(const Allocator& alloc = Allocator())
- 获取当前栈踪迹 (跳过指定数量的帧):
static std::basic_stacktrace current(size_t skip, const Allocator& alloc = Allocator())
- 迭代器: 提供了
begin()
,end()
,rbegin()
,rend()
等迭代器,方便遍历栈帧。 - 大小:
size()
返回栈帧的数量。 - 索引访问:
operator[]
允许按索引访问栈帧。 - 转换为字符串:
to_string()
方法可以将整个栈踪迹转换为易于阅读的字符串。
- 获取当前栈踪迹:
基本用法示例
C++
#include <iostream>
#include <stacktrace> // 引入 C++23 栈踪迹库void bar(int x) {std::cout << "Current stack trace in bar():\n";// 获取当前栈踪迹std::stacktrace st = std::stacktrace::current();std::cout << st << "\n"; // 使用默认的 ostream 输出// 或者手动迭代for (const auto& frame : st) {std::cout << " " << frame.description()<< " [" << frame.source_file() << ":" << frame.source_line() << "]\n";}
}void foo(int y) {bar(y * 2);
}int main() {std::cout << "Starting main...\n";foo(10);std::cout << "Exiting main.\n";return 0;
}
编译和运行注意事项:
为了获得最详尽的栈踪迹信息(如文件名、行号和未修饰的函数名),通常需要在编译时启用调试信息,并可能需要关闭一些优化。
- GCC/Clang: 使用
-g
标志。为了获得更清晰的函数名,有时可能需要链接时的一些选项,或者使用工具如addr2line
对输出的地址进行解析(尽管std::stacktrace
库致力于在内部处理这些)。 - MSVC: 使用
/Zi
或/Z7
标志。
输出可能如下所示 (具体格式和详细程度取决于编译器和平台):
Starting main...
Current stack trace in bar():
0# bar(int) at /path/to/your/source.cpp:8
1# foo(int) at /path/to/your/source.cpp:17
2# main at /path/to/your/source.cpp:22
3# ... (系统调用相关的帧)bar(int) [source.cpp:8]foo(int) [source.cpp:17]main [source.cpp:22]...
Exiting main.
与异常处理的集成
栈踪迹库与 C++ 的异常处理机制可以很好地结合。虽然标准异常类 std::exception
及其派生类本身并不直接携带栈踪迹信息(为了保持 ABI 兼容性),但开发者可以轻松地创建自定义异常类,在异常被抛出时捕获并存储栈踪迹。
P0881R7 的一个重要设计目标是与未来的提案(例如 P2370 “Stack trace from std::exception
”)协同工作,该提案旨在将栈踪迹更紧密地集成到标准异常类中。
当前的一个简单集成示例:
C++
#include <iostream>
#include <stacktrace>
#include <stdexcept>
#include <string>class traceable_error : public std::runtime_error {
public:traceable_error(const std::string& what_arg): std::runtime_error(what_arg), trace_(std::stacktrace::current(1)) {} // 跳过 traceable_error 构造函数本身const std::stacktrace& trace() const noexcept {return trace_;}private:std::stacktrace trace_;
};void function_c() {throw traceable_error("Something went wrong in function_c!");
}void function_b() {function_c();
}void function_a() {function_b();
}int main() {try {function_a();} catch (const traceable_error& e) {std::cerr << "Caught an exception: " << e.what() << "\n";std::cerr << "Stack trace:\n" << e.trace() << "\n";} catch (const std::exception& e) {std::cerr << "Caught a standard exception: " << e.what() << "\n";}return 0;
}
优势与价值
C++23 栈踪迹库的引入带来了诸多好处:
- 标准化与可移植性: 开发者不再需要为不同平台编写和维护特定的栈踪迹代码。
- 简化调试: 更容易理解程序在发生错误或崩溃时的执行路径。
- 增强的错误报告: 可以在日志或错误报告中包含详细的栈踪迹,帮助快速定位问题。
- 更佳的诊断工具: 为构建复杂的诊断和监控系统提供了基础。
- 代码可读性和维护性提升: 使用标准库特性通常比依赖外部或特定平台的方法更清晰。
潜在的考量
- 性能开销: 获取栈踪迹通常不是一个零开销的操作。虽然实现会尽力优化,但在性能敏感的代码路径中频繁调用
std::stacktrace::current()
可能需要谨慎评估其影响。通常,它主要用于错误处理和调试场景,而不是核心计算逻辑。 - 信息完整性: 栈踪迹的详细程度(例如,是否包含内联函数的帧,函数名的清晰度)仍然会受到编译器优化选项、调试信息生成和底层平台能力的影响。某些情况下,栈帧信息可能不完整或难以解析。
- 安全考量: 在某些安全敏感的应用中,暴露详细的栈踪迹信息(包括函数名和源文件路径)可能需要谨慎处理,以避免泄露内部实现细节。
总结
C++23 的 <stacktrace>
库是 C++ 语言在开发者体验和实用性方面迈出的重要一步。它提供了一个期待已久的标准化工具,用于捕获和处理调用栈信息,极大地简化了调试、错误诊断和日志记录等任务。虽然开发者仍需注意编译选项和潜在的性能影响,但这一新特性无疑将成为 C++ 开发者工具箱中的宝贵补充,帮助我们构建更健壮、更易于维护的应用程序。随着编译器对 C++23 的支持逐渐完善,我们期待栈踪迹库在实际项目中得到广泛应用。
相关文章:
C++23 新利器:深入解析栈踪迹库 (P0881R7)
文章目录 为何需要标准化的栈踪迹?P0881R7 的核心组件与使用基本用法示例与异常处理的集成优势与价值潜在的考量总结 对于 C 开发者而言,调试和错误诊断一直是开发周期中不可或缺但又充满挑战的一环。当程序崩溃或发生未预期行为时,获取清晰、…...
2025-05-06 事业-独立开发项目-记录
摘要: 2025-05-06 事业-独立开发项目-记录 独立开发项目记录 Product Hunt | InDev 独立开发者导航站https://www.producthunt.com/ Nomads.com - Best Places to Live for Digital Nomads (formerly Nomad List)https://nomads.com/ InDev 独立开发者导航站https://indev.bei…...
【Linux系统】探索进程等待与程序替换的奥秘
文章目录 前言一、重谈进程创建1.1 fork 函数1.2 写时拷贝1.3 fork 的常规用法1.4 fork 调用失败的原因1.5 创建一批进程 二、进程终止2.1 进程退出场景2.2 strerror 函数的作用2.3 errno 全局变量2.4 程序异常机制2.5 进程退出方式 三、进程等待3.1 进程等待必要性3.2 进程等待…...
Github 2025-05-06Python开源项目日报 Top10
根据Github Trendings的统计,今日(2025-05-06统计)共有10个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量Python项目10C++项目2TypeScript项目1系统设计指南 创建周期:2507 天开发语言:Python协议类型:OtherStar数量:241693 个Fork数量:42010 次…...
【愚公系列】《Manus极简入门》021-音乐创作助手:“音符魔术师”
🌟【技术大咖愚公搬代码:全栈专家的成长之路,你关注的宝藏博主在这里!】🌟 📣开发者圈持续输出高质量干货的"愚公精神"践行者——全网百万开发者都在追更的顶级技术博主! …...
【Azure Redis】Redis导入备份文件(RDB)失败的原因
问题描述 在测试Azure Redis的导入/导出备份文件的功能中,突然发现在Redis 4.0上导入的时候,一直报错。 image.png 问题解答 因为门户上只是显示导入失败,没有任何错误消息说明。根据常理推断,Redis 的RDB文件格式都具有一致性。居…...
git “分离头指针”(detached HEAD) 状态。
在 Git 中,当你运行 git branch 命令时,看到如下输出: * (detached from 5b596b5)master 其中的: * (detached from 5b596b5) 表示你当前处于 “分离头指针”(detached HEAD) 状态。 🧠 什…...
Gitee的介绍
目录 1.Gitee介绍: 1.1 代码托管 1.2 本土化优势 1.3 企业级服务 1.4 开源生态 1.5 多形态适配 定位:国内开发者首选的高效代码协作平台,兼顾个人开源与企业级私有开发需求。 2.Gitee和GitHub区别 3.Gitee使用教程 4.Gitee相关…...
NoUniqueKey问题和Regular join介绍
问题背景 在flink任务中,遇到了 NoUniqueKey Join的情况,导致了数据膨胀,和下游结果与数据库数据不一致问题 那NoUniqueKey Join为什么会导致问题呢,下面是其中一种场景示例: 为什么会出现 NoUniqueKey :…...
TC8:SOMEIP_ETS_027-028
SOMEIP_ETS_027: echoUINT8 目的 检查method方法echoUINT8的参数及其顺序能够被顺利地发送和接收 说白了就是检查UINT8数据类型参数在SOME/IP协议层的序列化与反序列化是否正常。 UINT8相比于测试用例SOMEIP_ETS_021: echoINT8中的SINT8数据类型来说,属于无符号整数,也就是…...
小微企业SaaS ERP管理系统,SpringBoot+Vue+ElementUI+UniAPP
小微企业的SaaS ERP管理系统,ERP系统源码,ERP管理系统源代码 一款适用于小微企业的SaaS ERP管理系统, 采用SpringBootVueElementUIUniAPP技术栈开发,让企业简单上云。 专注于小微企业的应用需求,如企业基本的进销存、询价&#…...
css filter 常用方法函数和应用实例
1. blur() 模糊 filter: blur(半径);参数:模糊半径(像素),值越大越模糊 示例:filter: blur(5px);2. brightness() 亮度 filter: brightness(百分比); 参数:1原始对比度,0全灰,>…...
chrome inspect 调试遇到的问题
1、oppp 手机打开webview 的时候, 报错这个并没有页面 Offline #V8FIG6SGLN75M7FY Pending authentication: please accept debugging session on the device. 解决方法,保持chrome 浏览器在显示的状态 去设置里开启usb 调试再关闭,反复重…...
Kotlin 中 List 和 MutableList 的区别
在 Kotlin 中,List 和 MutableList 是两种不同的集合接口,核心区别在于可变性。 Kotlin 集合框架的重要设计原则:通过接口分离只读(read - only)和可变(mutable)操作,以提高代码的安…...
openssl 生成自签名证书实现接口支持https
1.下载安装openssl Win32/Win64 OpenSSL Installer for Windows - Shining Light Productions 2.配置环境变量 将 openssl 的目录(D:\tools\openssl\bin)添加到 path 中 3.生成自签名证书 找一个存证书的目录打开powershell 3.1 生成私钥 openssl gen…...
React 中集成 Ant Design 组件库:提升开发效率与用户体验
React 中集成 Ant Design 组件库:提升开发效率与用户体验 一、为什么选择 Ant Design 组件库?二、基础引入方式三、按需引入(优化性能)四、Ant Design Charts无缝接入图标前面提到了利用Redux提供全局维护,但如果在开发时再自己手动封装组件,不仅效率不高,可能开发的组件…...
神经网络:节点、隐藏层与非线性学习
神经网络:节点、隐藏层与非线性学习 摘要: 神经网络是机器学习领域中一种强大的工具,能够通过复杂的结构学习数据中的非线性关系。本文从基础的线性模型出发,逐步深入探讨神经网络中节点和隐藏层的作用,以及它们如何…...
vue+tsc+noEmit导致打包报TS类型错误问题及解决方法
项目场景: 提示:这里简述项目相关背景: 当我们新建vue3项目,package.json文件会自动给我添加一些配置选项,这写选项基本没有问题,但是在实际操作过程中,当项目越来越复杂就会出现问题,本文给大家分享vuetscnoEmit导致打包报TS类型错误问题及…...
Ragflow服务器上部署教程
参考官方文档进行整理 克隆相应代码 git clone https://github.com/infiniflow/ragflow.git修改vm.max_map_count sudo sysctl -w vm.max_map_count262144修改 daemon.json文件 {"registry-mirrors": ["https://docker.m.daocloud.io","https://0…...
Ubuntu 系统中解决 Firefox 中文显示乱码的完整指南
Firefox 是一款流行的网络浏览器,但在 Ubuntu 系统中有时会遇到中文显示乱码的问题。本文将为您提供一个全面的解决方案,帮助您轻松解决这个烦人的问题。 问题概述 在 Ubuntu 系统中使用 Firefox 浏览器时,有时会发现中文字符显示为乱码或方块。这通常是由于缺少合适的中文…...
JVM——垃圾回收
垃圾回收 在Java虚拟机(JVM)的自动内存管理中,垃圾回收(Garbage Collection, GC)是其核心组件之一。它负责回收堆内存中不再使用的对象所占用的内存空间,以供新对象的分配使用。下面我们将深入探讨JVM中的…...
【AI News | 20250506】每日AI进展
AI Repos 1、gitsummarize GitSummarize是一个在线工具,用户只需将GitHub URL中的“hub”替换为“summarize”,即可为任何公开或私有代码库生成交互式文档。该工具利用Gemini分析代码结构,自动生成系统级架构概述、目录和文件摘要、自然语言…...
LabVIEW高冲击加速度校准系统
在国防科技领域,高 g 值加速度传感器广泛应用于先进兵器研制,如深侵彻系统、精确打击弹药及钻地弹药等。其性能指标直接影响研究结果的准确性与可靠性,因此对该传感器进行定期校准意义重大。高冲击加速度校准系统具备多方面功能,适…...
优化算法 - intro
优化问题 一般形式 minimize f ( x ) f(\mathbf{x}) f(x) subject to x ∈ C \mathbf{x} \in C x∈C 目标函数 f : R n → R f: \mathbb{R}^n \rightarrow \mathbb{R} f:Rn→R限制集合例子 C { x ∣ h 1 ( x ) 0 , . . . , h m ( x ) 0 , g 1 ( x ) ≤ 0 , . . . , g r …...
从PotPlayer到专业播放器—基于 RTSP|RTMP播放器功能、架构、工程能力的全面对比分析
从PotPlayer到专业播放器SDK:工程项目怎么选择合适的播放方案? ——基于 RTSP、RTMP 播放器功能、架构、工程能力的全面对比分析 在许多音视频项目早期,我们都听过这句话: “本地测试就用 PotPlayer 播吧,能播就行了…...
EasyRTC嵌入式音视频通信SDK技术,助力工业制造多场景实时监控与音视频通信
一、背景 在数字化时代,实时监控广泛应用于安防、工业、交通等领域。但传统监控系统实时性、交互性欠佳,难以满足需求。EasyRTC作为先进实时通信技术,具有低延迟、高可靠、跨平台特性,能有效升级监控系统。融入EasyRTC后…...
MPay码支付系统第四方聚合收款码多款支付插件个人免签支付源码TP8框架全开源
一、源码描述 这是一套码支付源码(MPay),基于TP8框架,前端layui2.9后端PearAdmin,专注于个人免签收款,通过个人的普通收款码,即可实现收款通知自动回调,支持绝大多数商城系统&#…...
wrod生成pdf。[特殊字符]改背景
import subprocess import os,time from rembg import remove, new_session from PIL import Image import io from docxtpl import DocxTemplate, InlineImage from docx.shared import Inches input_folder ‘tupian’ # 输入文件夹 kouchu_folder ‘kouchu’ # 去背景图像…...
动手学深度学习12.1. 编译器和解释器-笔记练习(PyTorch)
以下内容为结合李沐老师的课程和教材补充的学习笔记,以及对课后练习的一些思考,自留回顾,也供同学之人交流参考。 本节课程地址:无 本节教材地址:12.1. 编译器和解释器 — 动手学深度学习 2.0.0 documentation 本节…...
数字文明时代开源技术驱动的商业范式重构:基于开源AI大模型、AI智能名片与S2B2C商城小程序源码的协同创新研究
摘要:数字文明时代,数字技术正以指数级速度重构全球经济与社会结构。本文聚焦开源AI大模型、AI智能名片与S2B2C商城小程序源码的协同创新机制,从技术架构、商业逻辑、实践案例三个维度展开系统研究。基于多行业实证数据,揭示开源技…...
【Bootstrap V4系列】学习入门教程之 组件-轮播(Carousel)
Bootstrap V4系列 学习入门教程之 组件-轮播(Carousel) 轮播(Carousel)一、How it works二、Example2.1 Slides only 仅幻灯片2.2 With controls 带控制装置2.3 With indicators 带指示器2.4 With captions 带字幕 轮播࿰…...
嵌入式openharmony标准鸿蒙系统驱动开发基本原理与流程
第一:鸿蒙概述 OpenHarmony采用多内核(Linux内核或者LiteOS)设计,支持系统在不同资源容量的设备部署。当相同的硬件部署不同内核时,如何能够让设备驱动程序在不同内核间平滑迁移,消除驱动代码移植适配和维护的负担,是OpenHarmony驱动子系统需要解决的重要问题。 …...
Leetcode 刷题记录 08 —— 链表第二弹
本系列为笔者的 Leetcode 刷题记录,顺序为 Hot 100 题官方顺序,根据标签命名,记录笔者总结的做题思路,附部分代码解释和疑问解答,01~07为C语言,08及以后为Java语言。 01 合并两个有序链表 /*** Definition…...
PaddlePaddle 和PyTorch选择与对比互斥
你遇到的错误信息如下: RuntimeError: (PreconditionNotMet) Tensors dimension is out of bound.Tensors dimension must be equal or less than the size of its memory.But received Tensors dimension is 8, memorys size is 0.[Hint: Expected numel() * Size…...
极新月报·2025.4人工智能投融资观察
“ AI投资从‘量’向‘质’过渡 ” 4月重点关注: 1、四月人工智能领域投融资事件105起,披露金额78.63亿人民币。 2、亿级人民币以上金额的投资事件共20起 。 3、四月人工智能领域出现1起IPO事件。 4、在所有融资事件里,除去股权投资&…...
C++ vector 介绍与使用
目录 1.vector是什么? 2.vector的使用 2.1vector的构造函数 2.2vector iterator 的使用 2.3vector 空间增长问题 2.4vector的增删查改 1.vector是什么? 1. vector是表示可变大小数组的序列容器。 2. 就像数组一样,vector也 采用连续的存储…...
可以下载blender/fbx格式模型网站
glbxz.com glbxz.com可以下载blender/fbx格式模型。当然里面有免费的...
Vi/Vim 编辑器详细指南
Vi/Vim 编辑器详细指南 简介一、模式详解1. 命令模式(Normal Mode)2. 插入模式(Insert Mode)3. 可视模式(Visual Mode)4. 命令行模式(Ex Mode)二、核心操作1. 保存与退出2. 导航与移动3. 编辑与文本操作4. 搜索与替换三、高级技巧1. 多文件与窗口操作2. 宏录制3. 寄存器…...
LeetCode 热题 100 22. 括号生成
LeetCode 热题 100 | 22. 括号生成 大家好,今天我们来解决一道经典的算法题——括号生成。这道题在 LeetCode 上被标记为中等难度,要求生成所有可能的并且有效的括号组合。这是一道非常经典的回溯法题目,非常适合用来练习递归和回溯的技巧。…...
UE5 MetaHuman眼睛变黑
第5个材质MI_EyeOcclusion_Inst修改成透明即可...
【C语言】--指针超详解(一)
目录 一.内存和地址 1.1--内存 1.2--如何理解编址 二.指针变量和地址 2.1--取地址操作符(&) 2.2--指针变量和解引用操作符(*) 2.2.1--指针变量 2.2.2--如何理解指针类型 2.2.3--解引用操作符 2.3--指针变量的大小 三.指针变量类型的意义 3.1--从指针的解引用方…...
高频工业RFID读写器-三格电子
高频工业RFID读写器 型号:SG-HF40-485、SG-HF40-TCP 产品功能 高频工业读写器(RFID)产品用在自动化生产线,自动化分拣系统,零部件组装产线等情境下,在自动化节点的工位上部署RFID读写设备,通过与制品的交互…...
驱动开发系列57 - Linux Graphics QXL显卡驱动代码分析(四)显示区域绘制
一:概述 前面在介绍了显示模式设置(分辨率,刷新率)之后,本文继续分析下,显示区域的绘制,详细看看虚拟机的画面是如何由QXL显卡绘制出来的。 二:相关数据结构介绍 struct qxl_moni…...
6.5 行业特定应用:金融、医疗、制造等行业的定制化解决方案
金融、医疗和制造行业作为全球经济支柱,面临数据复杂性、实时性需求和严格合规性的共同挑战,同时各行业因业务特性衍生出独特需求。金融行业需应对市场波动、欺诈风险和多国法规,医疗行业聚焦精准诊断和患者数据隐私,制造业则强调…...
【Linux我做主】深入探讨从冯诺依曼体系到进程
从冯诺依曼体系到进程 从冯诺依曼体系到进程github地址1. 前言2. 计算机硬件2.1 冯诺依曼体系结构2.2 冯诺依曼模型的三大要点2.3 从QQ聊天认识:冯诺依曼体系下数据是如何流动的?发送方数据流动接收方数据流动 3. 计算机软件的根基——操作系统3.1 操作系…...
idea更换jdk版本操作
有时候我们有更换jdk版本的问题,自己电脑可能有多个版本,下面来介绍修改jdk版本修改修改什么地方 1 2 3 4 5 6 再修改pom即可,还有环境变量即可,希望有帮到大家!...
npm install下载插件无法更新package.json和package-lock.json文件的解决办法
经过多番查证,使用npm config ls查看相关配置等方式,最后发现全局的.npmrc文件的配置多写了globaltrue,去掉就好了 如果参数很多,不知道是哪个参数引起的,先只保留registryhttp://xxx/,试试下载࿰…...
机器学习实操 第二部分 神经网路和深度学习 第13章 使用TensorFlow加载和预处理数据
机器学习实操 第二部分 神经网路和深度学习 第13章 使用TensorFlow加载和预处理数据 内容概要 第13章深入探讨了如何使用TensorFlow加载和预处理数据。本章首先介绍了tf.data API,它能够高效地加载和预处理大规模数据集,支持并行文件读取、数据打乱、批…...
WebSoket的简单使用
一、WebSocket简介 1.1、双向通信/全双工 客户端和服务器之间同时双向传输,全双工通信允许客户端和服务器随时互相发送消息,不需等一方发送请求后另一方才进行响应。 适用要低延迟/实时交互的场景,如在线游戏、即时通讯、股票行情等。 1.2…...
01_线性表
一、线性表的顺序存储 逻辑上相邻的数据元素,物理次序也相邻。占用连续存储空间,用“数组”实现,知道初始位置就可推出其他位置。 00_宏定义 // 函数结果状态代码 #define TRUE 1 #define FALSE 0 #define OK 1 #define ERROR 0 #defin…...