总结下c/c++的一些调试经验
工作2年,干了一年ARM平台嵌入式,一年后台,总结下这两年开发中调试的经验。我把调试手段分成2种:打印日志和用工具分析。因为平时主要开发在Linux平台,就以GDB为例
一、打印日志
1. 合理设置日志级别
一般的日志库都会有日志级别,合理使用能避免因线上环境的日志打印过大而导致的磁盘占用过高,搜索日志耗时太长,因为日志大小超过设置值,导致一天打印日志数过多等问题。线上环境的日志一般需要以下级别:
Info,表示业务走到关键点时候的变量信息,队列消费数目,等待处理数目这些用于定位业务是否正常,分析程序性能所需要的指标信息
Warn,正常情况下程序不应该出现这种状态,但不是致命错误
Error,程序异常了,或者已经挂掉了
而测试环境,功能测试时可以把日志设置成Debug级别,最后代码提交时,可以带有Debug级别的日志,但务必保证留下的日志精简,明确。一个精简的调试日志,可以替代大部分函数内注释
2. 善于运用内置的调试宏
c++提供的几个宏:__FILE__ , __FUNCTION__ , __LINE__ 帮助定位代码位置
void Foo(){
//error here
Log.Error<<__FUNCTION__<<" is errors in Line["<<__LINE__<<"] varable is "<<var<<endl;
}
当然,有的库自带了这些信息
3. 日志划分
多线程下,一个模块可能有多个业务流程。日志划分时最好按照业务,或者模块划分。比如数据库操作的一个日志,和消息中间件通信的一个日志。便于日志归档管理
4. 避免滥用日志
日志级别设成info时,debug是不会打的。但这并不代表这条debug级别的日志不会消耗cpu的资源。看下面这个例子
log.debug("fun[%s] return [%d]", __func__, fun());
这条日志不会被打印,但仍然会执行,并且消耗cpu。理由有2点。
- 日志的接口函数被调用了,只是判断级别不够,不输出到日志文件。
- fun()作为函数的实参传递,fun先被掉用,再将结果传给日志接口
所以一定不能觉得测试的日志反正线上不会显示,就可以随意打印。
总结起来,日志需要注意以下3点
- 一条日志描述清楚when,what,where信息
- 在可能出现问题的地方打日志,通过其他日志能推断出信息的地方,无需再打日志
- 调用日志打印接口时,不要调用函数
二、GDB调试以及coredump分析
1. ELF文件
ELF(Executable Linkable Format)是COFF(Common File Format)的格式变种。系统中采用ELF的有以下几种
- 可重定位文件
- 可执行文件
- 共享目标文件
- 核心转储文件(Core Dump File)
这4类文件在Linux中可以通过`file [file name]`看出属于哪一种。核心转储文件就是我们常说的core文件,当程序意外终止时,系统将进程的所有地址空间以及终止信息存在该文件中。所以我们需要对ELF有所了解,才能正确分析core文件。有的系统限制了core文件的大小,需要ulimit -a看一下,然后设置成需要的值,例如ulimit -c unlimit
2.处理目标文件的工具
- AR:创建静态库,插入删除列出和提取成员
- STRIP:列出一个目标文件中所有可打印的字符串
- NM:列出一个目标文件的符号表中定义的符号
- SIZE:列出目标文件中节的名字和大小
- READELF:显示一个目标文件的完整结构,包括ELF头中编码的所有信息。包含SIZE和NM的功能
- OBJDUMP:所有二进制工具之母,能够显示一个目标文件中所有的信息。最大的作用是反汇编.text节中的二进制指令
- LDD:列出一个可执行文件在运行时需要的共享库
常用的参数举例:
readelf -h [file] 查看elf头部信息
readelf -S
objdump -d -j [section] [file]
3. gdb的使用

(图摘自CSAPP ch3.11)
产生coredump文件时分析步骤:
1. bt 查看程序crash位置,where也可以
2. `frame number` `up/down n` 到对应标号的栈帧
3. list + 查看代码
4. info locals 简化命令i locals。看栈变量、函数行
5. `print name_of_variable` or `p name_of_variable` 打印变量
需要注意的是,由于现在gcc编译器普遍开了优化,使源代码和生成的代码之间的映射更难看出。这些优化对程序性能有提升,却增加了调试的难度。当看不出时,需要结合日志和代码推断问题。
总结下c/c++的一些调试经验的更多相关文章
- (转)CMOS Sensor的调试经验分享
CMOS Sensor的调试经验分享 我这里要介绍的就是CMOS摄像头的一些调试经验. 首先,要认识CMOS摄像头的结构.我们通常拿到的是集成封装好的模组,一般由三个部分组成:镜头.感应器和图像信号处 ...
- CMOS Sensor的调试经验分享
转自:http://bbs.52rd.com/forum.php?mod=viewthread&tid=276351 CMOS Sensor的调试经验分享 我这里要介绍的就是CMOS摄像头的一 ...
- 调试经验--硬盘U菜
调试经验--硬盘U菜 随着嵌入式设备功能的开发,随着对存储设备的需求:需要存储大量数据信息.需要在转储数据,U盘升级功能等. 在使用存储设备的过程中,我们遇到一些问题,也总结了些经验: 1.几 ...
- CMOS Sensor的调试经验分享【转】
转自:https://blog.csdn.net/yapingmcu/article/details/37817727 转自:http://bbs.52rd.com/forum.php?mod=vie ...
- 总结OpenvSwitch的调试经验
相信相信不少人在开始用OpenvSwitch(以下简称OVS)或者修改源码的时候,都不知道如何去调试,根据我的学习历程以及从网上搜集的资料做一个汇总. 一. 个人经验 从网上找相关资料,熟悉Openv ...
- 程序异常崩溃后用windbg辅助调试解决的经验 以及 堆栈问题调试经验
1,程序异常崩溃后用windbg辅助调试解决的经验 状况:我的程序调用别人的库做 文件写入工作. 在这一过程中出现异常,程序崩溃. 经反复检查,认为自己的程序没有错,但无法判断在别人库里哪里有错. ...
- 17调试经验之串口读写flash协议
一是设计功能 我的理解协议就是一个命令包,通过给出不同的控制命令,来调动不同的功能模块,实现不同的功能,如读数据,写数据,擦除等. 二设计过程 先看了尤老师的视频,主要讲了大致设计原理和总体框架,当然 ...
- 电机三环pid控制及调试经验
一.伺服电机的双环pid 双环pid在正常底盘运动的控制中已经足够了,但是对于双轴云台的控制来说,双环pid的云台控制的响应速度是远远不够的,所以加入了电流环的控制. 两篇大佬的文章--这是我学习pi ...
- win10 下visual studio 2015 在调试模式下不能跟踪源文件
win10 下visual studio 2015 在调试模式下不能跟踪源文件,只要一调试就会关闭(隐藏)打开的文档,非常不方便.经过一番折腾,发现是配置的问题. 如果安装多个版本的VS,请删除对应版 ...
随机推荐
- eatwhatApp开发实战(六)
上次,我们为app添加了本地存储的功能,但会发现一但退出app则存储的商家集合就消失,但其实本地已经存储了记录只是没去读取罢了. 接下来我们来实现这个功能. /** * 获取本地数据 */ priva ...
- [PHP插件教程]003.PhpRedis
PhpRedis 介绍 Mac安装步骤 安装Redis 安装PhpRedis 示例代码 介绍 Redis是一个高性能的key-value数据库. Redis提供了Java,C/C++,C#,PHP,J ...
- Shell概述1
Shell概述1 脚本文件内容(vim ex2) #!/bin/bash #If no arguments,then listing the current directory. #Otherwise ...
- Matlab矩阵学习二 矩阵的修改
Matlab矩阵的修改 一.元素修改 (1).矩阵扩充 (2)矩阵删除某行或某列 删除某行:A(m,:)=[] %删除A矩阵的第m行 删除某列: A(:,n)=[] %删除A矩阵的第n列 ...
- 关于Backus-Naur Form巴克斯诺尔范式和扩展巴克斯范式的知识点和相关词语中英文对照
巴克斯诺尔范式的相关词语中英文对照和知识点 syntax 语法 强调的是编程语言的组形式,例如一个句子中会包含表达式.陈述还有各种单元等等 semantics 语义 强调的是这个编程语言的实际含义,例 ...
- Spring boot Sample 006之spring-boot-custom-servlet
一.环境 1.1.Idea 2020.1 1.2.JDK 1.8 二.步骤 2.1.点击File -> New Project -> Spring Initializer,点击next 2 ...
- 开心一下-实现一个基于Java的中文编程语言2
https://mp.weixin.qq.com/s/VmCTvh0c7X9DjIgIMycdlw 上一篇所提到的只是使用中文写Java,而不能算作一门新的语言.作为一门中文语言,需要语言提供的关 ...
- jchdl - GSL实例:FullAdder
https://mp.weixin.qq.com/s/CtT08xZON0YxnheqDM2FAw 全加器是能够计算低位进位的二进制加法电路.与半加器相比,全加器不只考虑本位计算结果是否有进位,也考虑 ...
- DataGuard VS Beedup & GoldenGate灾备方案参数对比
世上本无完美产品,只有合适的才是最好的! 用户重视灾备数据站点的建设,毋庸置疑必备品.如果考虑带宽及事务完整性保证,存储灾备和操作系统级灾备局限性显而易见. 商用价值一般用于解决数据库自带辅助功能的短 ...
- MethodHandle(方法句柄)系列之二:方法句柄的简单使用
二话不说,上代码 /** * * @author LiuYeFeng<897908343@qq.com> * @date 2015年4月8日 下午10:41:13 * @CopyRigh ...