异常处理/__LINE__ 与 __FILE__ 宏在调试和异常处理中的高级使用

文章目录

  • 概述
  • 痛点分析
  • _LINE_ 代码所在行号
    • _LINE_ 直接转为字符串
    • _LINE_ 作为整型数据使用
    • _LINE_标记宏函数的调用位置
  • _FILE_ 代码所在文件名
    • 简单实验
    • 不期望 _FILE_ 宏代表全路径
  • assert 使用了 _FILE_ 和 _LINE_
  • 借助TLS技术
  • 小结

概述

_LINE_和_FILE_是C/C++中的预定义宏,分别用于获取当前代码所在的行号和文件名。本文详细讲述了它们的使用场景和使用方法,消除了一些陈旧错误的理解,重点实践了它们在软件调试、系统异常处理过程中举足轻重的作用。
编译器在预处理阶段将它们替换为相应的值,具体来说:_LINE_宏会被替换为当前代码所在的行号,表示该行号在源文件中的位置;_FILE_宏会被替换为当前代码所在的文件名,表示包含该代码的源文件的文件名。通过在日志记录或错误消息中使用_LINE_宏,可以动态地获取代码中出现问题的位置,并将其包含在日志或错误消息中,以帮助开发人员在调试和错误处理中排查问题。要再次强调的是,这些宏的值是在编译时确定的,因此它们提供的行号和文件名是编译时的信息,而不是运行时的信息。

转载请标明原文链接,
https://blog.csdn.net/quguanxin/category_6223029.html

痛点分析

#define STRINGIFY(x) #x          //it won't take effect, if you just do this
#define TOSTRING(x) STRINGIFY(x) //it works
//
std::string generateErrorMessage(std::string msg) {return msg + " in " + __FILE__ + " at line " + /*std::to_string(__LINE__)*/ TOSTRING(__LINE__);
}
//
int main() {std::string errorMsg = generateErrorMessage("Something went wrong.");std::cout << errorMsg.c_str() << std::endl;system("pause"); return 0;
}

执行结果,
在这里插入图片描述
如上,我们期望标记的是第13行代码位置,而运行结果却只能是 LINE 出现的第9行代码。FILE 存在相似的情况。

LINE 代码所在行号

早些年由于对_LINE_宏的不够了解,以为,既然它是编译时确定的,那么就只能死板的在待调试行上写包含_LINE_宏的代码,而不能封装或传递它。而实际上,配合宏函数的使用或者将其作为size_t 类型的函数实参,可以非常灵活的使用它。_LINE_宏可以直接转为字符串、可以作为int类型使用、也可以封装在宏函数内部(最终客户端调用宏函数,LINE 被转换为宏函数所在的行),接下来我们展开论证。

LINE 直接转为字符串

#include <iostream>
#include <string>#define STRINGIFY(x) #x          //it won't take effect, if you just do this
#define TOSTRING(x) STRINGIFY(x) //it works// 定义一个宏来生成带有错误消息、文件名和行号的字符串
#define ERROR_MSG(msg) ("Error: " + std::string(msg) + " in " + __FILE__ + " at line " + TOSTRING(__LINE__))int main() {std::string errorMsg = ERROR_MSG("Something went wrong.");std::cout << errorMsg << std::endl;return 0;
}

在这里插入图片描述
在这里插入图片描述
(操作符#)是C/C++中的预处理器操作符,称为字符串化操作符(stringify operator)。在预处理阶段,它可以将宏参数或标识符转换为字符串常量。具体来说,#操作符会在宏展开过程中将其后面的标识符或参数转换为一个以双引号括起来的字符串。

LINE 作为整型数据使用

我们以 ros2/rcutils/error_handling.h中的函数为例,

/// Set the error message, as well as the file and line on which it occurred.
/*** This is not meant to be used directly, but instead via the* RCUTILS_SET_ERROR_MSG(msg) macro.** The error_msg parameter is copied into the internal error storage and must* be null terminated.* The file parameter is copied into the internal error storage and must* be null terminated.** \param[in] error_string The error message to set.* \param[in] file The path to the file in which the error occurred.* \param[in] line_number The line number on which the error occurred.*/
RCUTILS_PUBLIC void rcutils_set_error_state(const char * error_string, const char * file, size_t line_number);/// Set the error message, as well as append the current file and line number.
/*** If an error message was previously set, and rcutils_reset_error() was not called* afterwards, and this library was built with RCUTILS_REPORT_ERROR_HANDLING_ERRORS* turned on, then the previously set error message will be printed to stderr.* Error state storage is thread local and so all error related functions are* also thread local.** \param[in] msg The error message to be set.*/
#define RCUTILS_SET_ERROR_MSG(msg) \do {rcutils_set_error_state(msg, __FILE__, __LINE__);} while (0)

如上,首先在编译预处理阶段,LINE 被替换为 RCUTILS_SET_ERROR_MSG 宏函数的代码行号,然后其作为一个整型数据,也即作为 rcutils_set_error_state 函数 line_number 参数的实参被传递。接下来我们定义一个简单的可接收 _FILE_, __LINE__实参的函数,

//形参类型分别对应 const char* 和 size_t 
std::string generateErrorMessage(std::string msg, const char* file, size_t line_no) {return msg + " in " + file + " at line " + std::to_string(line_no);
}
//
int main() {std::string errorMsg = generateErrorMessage("Something went wrong.", __FILE__, __LINE__);std::cout << errorMsg << std::endl;system("pause");  return 0;
}

在这里插入图片描述
执行结果如上,LINE 被识别为其出现位置所在的行号。这种方案很好理解,但在实际使用中每次都要去传递 FILELINE 数据,让人感觉不是很舒服。一种更高级的办法是,将上述 generateErrorMessage 用宏函数封装。具体我们看下一小节。

_LINE_标记宏函数的调用位置

一个应对策略就是,定义宏函数,

//
#define ERROR_MSG(errorMsg, msg) \
{ \errorMsg = generateErrorMessage("Error:" + std::string(msg), __FILE__, __LINE__); \
}  \
//一种更优雅的写法是
#define ERROR_MSG(errorMsg, msg) \
do { errorMsg = generateErrorMessage("Error:" + std::string(msg), __FILE__, __LINE__); } while (0) 

C/C++语法要求在宏展开时,宏展开的结果必须是一个完整的语句。故在使用宏定义时,通常使用do {…} while(0)的技巧可以确保宏的语法完整性,使其在被展开时能够像代码块一样使用,且可以避免语法错误,提高代码可读性。当然,你也可以仅使用花括号。
在这里插入图片描述
测试结果如下,
在这里插入图片描述
如上测试结果表明,ERROR_MSG函数中无论_LINE_出现在其中的第几行,都无关紧要,_LINE_标记的是 ERROR_MSG 宏的调用位置(如上图行号为27行),而不是 _LINE_标记的直接位置 (如上图行号22行)。至此,算是消除了对 _LINE_的一个大误会,它的使用方法,远比我之前以为的要灵活。

FILE 代码所在文件名

在前文讲述 LINE 宏的过程中,也同时完成了 FILE 宏的使用实践,它是一个字符串常量,表示当前源文件的文件名,包括文件的路径,其对应的数据类型是 const char* ,也即 C 语言字符串。

简单实验

#include <stdio.h>
#include <iostream>
//
int main() {printf("当前源文件名:%s\n", __FILE__);system("pause");  return 0;
}

在这里插入图片描述
如上,输出 FILE 所在的代码文件的全路径。当资源很紧张,或者文件路径较深的时候,全路径名就会很烦人,咋办?

不期望 FILE 宏代表全路径

如上一小节,在Windows上使用 _FILE_ 宏,默认情况下其代表的是源代码文件的全路径名称,但在大多情况下,这会显得有点冗余、浪费资源。一般情况在同一个项目下,存在同名文件的可能性不大,同名且内容相同的可能更是不存在,因此我们仅保留文件名就可以。为此我们对 _FILE_ 宏进行如下重定义,

#ifdef _WIN32
#define FILE_NAME (strrchr(__FILE__, '\\') ? strrchr(__FILE__, '\\') + 1 : __FILE__)
#else
#define FILE_NAME (strrchr(__FILE__, '/') ? strrchr(__FILE__, '/') + 1 : __FILE__)
#endif

定义一个名为FILE_NAME的宏,它使用strrchr函数来查找最后一个斜杠字符’/',并返回该字符后面的字符串部分。如果没有斜杠字符,则直接返回__FILE__宏的值。要注意的是,在不同操作系统上,文件路径使用不同的分隔符。对比效果如下,清澈了许多,
在这里插入图片描述
测试用的源代码如下,

#include <stdio.h>
#include <iostream>
//
#ifdef _WIN32
#define FILE_NAME (strrchr(__FILE__, '\\') ? strrchr(__FILE__, '\\') + 1 : __FILE__)
#else
#define FILE_NAME (strrchr(__FILE__, '/') ? strrchr(__FILE__, '/') + 1 : __FILE__)
#endif
//
int main() {printf("当前源文件名:%s\n", FILE_NAME);system("pause"); return 0;
}

assert 使用了 FILELINE

int main() {printf("当前源文件名:%s, 当前代码行号:%s\n", FILE_NAME, __LINE__);assert(0 == 1);system("pause"); return 0;
}

在这里插入图片描述
通过上述测试,可以猜测,assert 内部极有可能是封装了 FILELINE 预处理宏的。看一下源码,

#ifdef NDEBUG#define assert(expression) ((void)0)
#else_ACRTIMP void __cdecl _wassert(_In_z_ wchar_t const* _Message,_In_z_ wchar_t const* _File,_In_   unsigned       _Line);#define assert(expression) (void)(                                                       \(!!(expression)) ||                                                              \(_wassert(_CRT_WIDE(#expression), _CRT_WIDE(__FILE__), (unsigned)(__LINE__)), 0) \)
#endif

与我们前面自定义函数的使用方法一致,先以 char* 和 int 为参数类型,定义函数,然后,使用宏封装此函数,这样,LINEFILE 就可以用来代表宏函数(assert) 的调用位置。

借助TLS技术

参见 src/error_handling.c 中的实现,定义线程本地存储TLS变量,

// g_ is to global variable, as gtls_ is to global thread-local storage variable
RCUTILS_THREAD_LOCAL bool gtls_rcutils_thread_local_initialized = false;
RCUTILS_THREAD_LOCAL rcutils_error_state_t gtls_rcutils_error_state;
RCUTILS_THREAD_LOCAL bool gtls_rcutils_error_string_is_formatted = false;
RCUTILS_THREAD_LOCAL rcutils_error_string_t gtls_rcutils_error_string;
RCUTILS_THREAD_LOCAL bool gtls_rcutils_error_is_set = false;

到这里就有点偏了… 已经超出了这个小主题… 跑到错误处理中了…

小结

在《异常处理/分析ROS2异常处理的设计和实现思路》(尚未发布)一文中,有提到过,针对调试信息,越直接越好,而 LINEFILE 宏所表现出来的,几乎就是最直接的。
站在开发者的角度上,无论是何种形式的异常处理,都是手段,我们根本目的始终是快速定位程序运行过程中的问题,并尽力地使其从问题中恢复正常运行。不同于此的,用户角度,作为软件的使用者,用户希望看到的告警信息应该是,可读性强、及时性好、清晰明了、具体详细的,并且好的告警信息不仅指出问题,还应该提供解决方案或建议,是可以操作和控制的。用户绝对不希望看到晦涩难懂的告警信息,而是希望能够快速地理解问题所在,因此给用户的告警信息必须是能简练和准确描述问题本质和原因的。虽说是具体详细,但也绝不是详细到哪个代码行,这是很容易理解的。用户期望了解的问题的具体细节,应该是业务层级的,操作层级的,而不是系统实现层级。

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

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

相关文章

visual studio使用结巴分词

1.安装Jieba.NET的NuGet程序包 2.主程序代码 using JiebaNet.Segmenter; using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks;namespace JiebaDemo {internal class Program{static void Main(string[] ar…

Qt---窗口系统

一、QMainWindow 1. 菜单栏(最多有一个) QMenuBar *bar MenuBar(); SetMenuBar(bar); QMenu *fileMenu bar->addMenu("文件"); 创建菜单 QAction *newAction fileMenu->addAction("新建"); 创建菜单项 添加分割线fileMenu-…

探索震坤行API:一键解锁高效工业用品采购新纪元!

震坤行是一家专注于工业用品的B2B电商平台&#xff0c;为企业客户提供一站式的工业用品采购服务。虽然震坤行没有直接公开通用的API接口供开发者调用&#xff0c;但通常大型企业或合作伙伴之间可以通过API进行系统集成和数据交互。以下是一个假设性的震坤行API接口调用示例与代…

车辆管理|基于SprinBoot+vue的4S店车辆管理系统(源码+数据库+文档)

4S店车辆管理系统 目录 基于SprinBootvue的4S店车辆管理系统 一、前言 二、系统设计 三、系统功能设计 系统实现 1管理员功能模块 2销售员功能模块 3维修员功能模块 四、数据库设计 五、核心代码 六、论文参考 七、最新计算机毕设选题推荐 八、源码获取&#xf…

小猪APP分发:一站式托管服务与高效应用分发解决方案

在当今快节奏的移动应用市场中&#xff0c;开发者不仅要专注于产品的创新与优化&#xff0c;还需面对复杂的应用发布流程与激烈的市场竞争。幸运的是&#xff0c;像小猪APP分发www.ppzhu.net这样的专业平台应运而生&#xff0c;它不仅解决了开发者在应用托管与分发上的诸多痛点…

Visual Studio,第1个hello world,入门C++,分别编译一个可以在Windows和Linux下运行的程序

本人的VxTerm&#xff0c;是在Visual Studio 2022下编写的。 其它的语言工具是不是也可以那么方便的使用&#xff0c;本人并不得而知&#xff0c;至少本人能知道&#xff1a;对于我来说&#xff0c;Visual Studio可以让我觉得C/C语言非常简单&#xff01; 一、安装Visual Stu…

linux性能监控之slabtop

slabtop命令是以实时的方式显示内核slab缓冲区的细节信息&#xff0c;是linux自带的命令 [rootk8s-master ~]# slabtop --helpUsage:slabtop [options]Options:-d, --delay <secs> delay updates-o, --once only display once, then exit-s, --sort <char&…

Maven 插件使用

1.spring-boot-maven-plugin 我们直接使用 maven package &#xff08;maven自带的package打包功能&#xff09;&#xff0c;打包Jar包的时候&#xff0c;不会将该项目所依赖的Jar包一起打进去&#xff0c;在使用java -jar命令启动项目时会报错&#xff0c;项目无法正常启动。…

代码随想录——二叉树的层序遍历Ⅱ(Leetcode107)

题目链接 层序遍历&#xff08;队列&#xff09; /*** Definition for a binary tree node.* public class TreeNode {* int val;* TreeNode left;* TreeNode right;* TreeNode() {}* TreeNode(int val) { this.val val; }* TreeNode(int val, Tre…

科技查新中医学科研项目查新点如何确立与提炼?案例讲解

一、前言 医学科技查新包括立项查新和成果查新两个部分&#xff0c;其中医学立项查新&#xff0c;它是指在医学科研项目申报开题之前&#xff0c;通过在一定范围内进行该课题的相关文献检索 ( 可以根据项目委托人的具体要求&#xff0c;进行国内检索或者进行国外检索 ) &#x…

介绍下InnoDB的锁机制?

在InnoDB中&#xff0c;锁可以分为两种级别&#xff0c;一种是共享锁&#xff08;S锁&#xff09;&#xff0c;另一种是排他锁&#xff08;X锁&#xff09;。 共享锁&排他锁 共享锁又称为读锁&#xff0c;由读取操作创建。其他用户可以并发读取数据&#xff0c;但直到所有…

能远程一起观看电影和直播的SyncTV

什么是 SyncTV &#xff1f; SyncTV 是一个允许您远程一起观看电影和直播的程序。它提供了同步观看、剧院和代理功能。使用 SyncTV&#xff0c;您可以与朋友和家人一起观看视频和直播&#xff0c;无论他们在哪里。SyncTV 的同步观看功能确保所有观看视频的人都在同一点上。这意…

C++ BuilderXE 计算程序运行时间精确到毫秒

#include <time.h> // //计算时间 clock_t start,end,dtStart; startclock(); // ProgressBar1->Percent0; // // ProgressBar1->Percenti/DDnum*100; // Application->ProcessMessages(); // //操作完成计时 …

使用Flask构建POST请求的Web应用

文章目录 准备工作创建路由处理POST请求创建表单页面运行应用结论 在Web开发中&#xff0c;处理POST请求是一项常见任务&#xff0c;特别是在构建表单提交、用户注册和数据提交等功能时。Flask是一个简单而强大的Python Web框架&#xff0c;它提供了方便的工具来处理HTTP请求&a…

目标检测算法YOLOv7简介

YOLOv7由Chien-Yao Wang等人于2022年提出&#xff0c;论文名为&#xff1a;《YOLOv7: Trainable bag-of-freebies sets new state-of-the-art for real-time object detectors》&#xff0c;论文见&#xff1a;https://arxiv.org/pdf/2207.02696 &#xff0c;项目网页&#xff…

激光雷达:盲人世界的导航灯塔

在科技日新月异的今天&#xff0c;一项名为“蝙蝠避障”的创新成果&#xff0c;正悄然改变着盲人朋友的日常生活&#xff0c;特别是在出行这一领域&#xff0c;它的应用如同一束光&#xff0c;照亮了前行的道路。本文将深入探讨激光雷达技术对盲人的帮助&#xff0c;揭示这项高…

【JavaWeb】网上蛋糕商城后台-商品管理

概念 本文讲解和实现网上蛋糕商城的后台管理系统中的商品管理功能。 商品列表 点击后台管理系统的head.jsp头部的“商品管理”功能选项&#xff0c;向服务器发送请求/admin/goods_list 因此需要在servlet包中创建AdminGoodsListServlet类&#xff0c;用于获取商品信息列表 …

【赠书活动第4期】《Rust编程与项目实战》

赠书活动 《Rust编程与项目实战》免费赠书 3 本&#xff0c; 收到赠书之后&#xff0c;写一篇 本书某一节内容 的学习博客文章。 可在本帖评论中表示参加&#xff0c;即可获得赠书&#xff0c;先到先得。学习心得博客链接&#xff0c;后面有空发上来。 赠书截止日期为送出3…

elementui的table行展开,左侧的icon有的需要有的不需要

百度了一些方法&#xff0c;都不好用&#xff0c;最后还是纯css解决&#xff0c;以下是效果&#xff1a; 代码实现&#xff1a; :deep(.el-table__row:nth-child(1) .el-table__expand-icon){ display: none; }

《解锁高效合同管理系统:优化业务流程,提升管理效率》

随着企业规模的扩大和业务复杂性的增加&#xff0c;合同管理变得愈发重要。合同是企业与客户、供应商、合作伙伴之间的法律约束和商业承诺&#xff0c;而有效的合同管理系统则成为企业提高运营效率、降低风险的关键工具。本文将探讨合同管理系统的重要性以及如何利用合同管理系…