《软件调试的艺术》笔记--停下来环顾程序

2024-04-14 21:48

本文主要是介绍《软件调试的艺术》笔记--停下来环顾程序,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.断点列表

创建的每个断点(包括断电、监视点和捕获点)都标识为从1开始的唯一整数标识符。这个标识符用来执行该断点上的各种

操作。调试器还包含一种列出所有断点及其属性的方法。

调试下面的代码:(代码1)

#include <stdio.h>void display(int i)
{i = i + 1;printf("i = %d\n",i);
}int main(void)
{int i = 1;display(i);return 0;
}
设置断点--显示断点列表--删除断点--显示删除后断点列表:

(gdb) break main
Breakpoint 1 at 0x80483d4: file a.c, line 11.
(gdb) break display
Breakpoint 2 at 0x80483aa: file a.c, line 5.
(gdb) info breakpoints
Num     Type           Disp Enb Address    What
1       breakpoint     keep y   0x080483d4 in main at a.c:11
2       breakpoint     keep y   0x080483aa in display at a.c:5
(gdb) delete 1
(gdb) info breakpoints
Num     Type           Disp Enb Address    What
2       breakpoint     keep y   0x080483aa in display at a.c:5
(gdb)

info breakpoints命令的简写为i b

2.设置断点

GDB中有许多指定断点的方式,下面是一些常见的方法。持久断点使用break命令(简写为b)
以下使用代码1调试:
在函数的入口处设置断点:
(gdb) break main
Breakpoint 1 at 0x80483d4: file a.c, line 11.

在当前活动源代码文件的某行设置断点:
(gdb) break 11
Breakpoint 3 at 0x80483d4: file a.c, line 11.

在某文件的某行处设置断点:
(gdb) break a.c:11
Note: breakpoint 3 also set at pc 0x80483d4.
Breakpoint 4 at 0x80483d4: file a.c, line 11.

在某文件的某函数入口处设置断点:
(gdb) break a.c:main
Breakpoint 6 at 0x80483d4: file a.c, line 11.

当设置一个断点时,该断点的有效性会持续到删除,禁用或退出GDB时,而临时断点在首次到达后就会被自动删除的断点。
临时断点使用tbreak命令(简写为tb)设置。它与break的用法相同。

2.断点的持久性

如果我们在修改和重新编译代码时没有退出GDB,那么下次在执行GDB的run命令时,GDB会感知到代码已修改,并重新
加载新版本:
以下使用代码1调试:
在A窗口执行如下:
(gdb) b main
Breakpoint 1 at 0x80483d4: file a.c, line 11.
(gdb) 

在B窗口执行如下:
修改display函数中的
i = i + 1;
i = i + 2;
并重新进行编译

回到A窗口继续执行:
(gdb) r
Breakpoint 1, main () at a.c:11
11              int i = 1;
(gdb) n
12              display(i);
(gdb) s
display (i=1) at a.c:5
5               i = i + 2;
发现已经是修改后的代码。

3.删除断点

以下使用代码1调试:
删除指定断点
(gdb) info breakpoints
Num     Type           Disp Enb Address    What
4       breakpoint     keep y   0x080483d4 in main at a.c:11
5       breakpoint     keep y   0x080483aa in display at a.c:5
(gdb) delete 4
(gdb) info breakpoints
Num     Type           Disp Enb Address    What
5       breakpoint     keep y   0x080483aa in display at a.c:5
(gdb)

删除所有断点
(gdb) info breakpoints
Num     Type           Disp Enb Address    What
5       breakpoint     keep y   0x080483aa in display at a.c:5
6       breakpoint     keep y   0x080483d4 in main at a.c:11
(gdb) delete
Delete all breakpoints? (y or n) y
(gdb) info breakpoints
No breakpoints or watchpoints.
(gdb)
其中delete命令的简写为d。

如果clear不带任何参数,则删除当前停止的断点:
(gdb) info breakpoints
Num     Type           Disp Enb Address    What
17      breakpoint     dis  y   0x080483d4 in main at a.c:11
19      breakpoint     keep y   0x080483aa in display at a.c:5
(gdb) r
Starting program: /root/a.out
Breakpoint 17, main () at a.c:11
11              int i = 1;
(gdb) clear
Deleted breakpoint 17
(gdb) info breakpoints
Num     Type           Disp Enb Address    What
19      breakpoint     keep y   0x080483aa in display at a.c:5
如果clear带参数,则删除所在行的所有断点。gdb允许在同一行设置多个断点,如果使用delete,必须一个一个删除,而clear
可以一起删除。
可以使用如下任何一种方式:
clear function、clear filename:function、clear line_number和clear file:linenumber
(gdb) info breakpoints
Num     Type           Disp Enb Address    What
14      breakpoint     keep y   0x080483d4 in main at a.c:11
15      breakpoint     keep y   0x080483d4 in main at a.c:11
16      breakpoint     keep y   0x080483d4 in main at a.c:11
(gdb) clear main
Deleted breakpoints 14 15 16
(gdb) info breakpoints
No breakpoints or watchpoints.
(gdb)

4.禁用断点

每个断点都可以被禁用或启用。只有当GDB遇到启用断点时才会暂停程序的执行;否则会忽略禁用的断点。
使用disable breakpoint-list命令禁用断点,使用enable breakpoint-list命令启用断点。
以下使用代码1调试:
(gdb) info breakpoints
Num     Type           Disp Enb Address    What
17      breakpoint     keep y   0x080483d4 in main at a.c:11
18      breakpoint     keep y   0x080483aa in display at a.c:5
(gdb) disable 17
(gdb) info breakpoints
Num     Type           Disp Enb Address    What
17      breakpoint     keep n   0x080483d4 in main at a.c:11
18      breakpoint     keep y   0x080483aa in display at a.c:5
(gdb) enable 17
(gdb) info breakpoints
Num     Type           Disp Enb Address    What
17      breakpoint     keep y   0x080483d4 in main at a.c:11
18      breakpoint     keep y   0x080483aa in display at a.c:5
(gdb)

还有一个enable once命令,在断点下次引起GDB暂停执行后被禁用,语法如下:
enable once breakpoint-list
对于下面的代码进行调试:(代码2)
#include <stdio.h>int main(void)
{int m = 1;int i;for (i=0; i<10; i++) {printf("%d",i);}return 0;
}
(gdb) b 8
Breakpoint 2 at 0x80483c5: file b.c, line 8.
(gdb) info breakpoints
Num     Type           Disp Enb Address    What
2       breakpoint     keep y   0x080483c5 in main at b.c:8
(gdb) disable 2
(gdb) enable once 2
(gdb) info breakpoints
Num     Type           Disp Enb Address    What
2       breakpoint     dis  y   0x080483c5 in main at b.c:8
(gdb) r
Starting program: /root/a.out
Breakpoint 2, main () at b.c:8
8                       printf("%d\n",i);
(gdb) c
Continuing.
0123456789
Program exited normally.
断点只执行了执行了一次,后面的断点被忽略。
没有disable once 命令。

5.断点属性

可以使用info breakpoints(简写为 i b)命令来获取设置的所有断点的清单以及它们的属性。
(gdb) i b
Num     Type           Disp Enb Address    What
1       breakpoint     keep y   0x080483d4 in main at a.c:11
2       breakpoint     keep y   0x080483aa in display at a.c:5

下面说明下i b的输出。
Num:标识符,断点的唯一标识符。
Type:类型,这个字段之处该断点是断点、监视点还是捕获点。
Disp:部署,每个断电都有一个部署,只是断点下次引起GDB暂停程序的执行后该断点上会发生什么事情。可能的部署有
一下3种。
keep:保持,下次达到断点后不改变断点。这时新建断点后的默认部署。
del:删除,下次达到断点后删除该断点。使用rbreak命令会出现该部署。
dis:禁用,下次达到断点时会禁用该断点。使用enable once命令会出现该部署。
Enb:启用状态,说明断点当前是启用还是禁用的。
Address:这时内存中设置断点的位置。
what:位置,显示了断点所在位置的行号和文件名。

6.恢复执行

恢复执行的方法有3类,第一类是使用step和next但不调试程序,仅执行代码的下一行然后再次暂停。第二类由使用continue
组成,使GDB无条件恢复程序的执行,直到它遇到另一个断电或者程序结束。最后一类方法涉及条件:用finish或until命令恢复。
这种情况下,GDB恢复执行;程序继续运行直到遇到某个预先确定的条件,到达另一个断点或者程序完成。
step(简写为s)、next(简写为n)和continue(简写为c)比较简单,不详细说明。有一点需要注意,next和step都采用一个可选的
数值参数,表示使用next或者step执行的额外行数,比如next 3在一行中键入next三次。

6.1.finish
finish(简写为fin)指示GDB恢复执行,直到敲好在当前栈帧完成之后为止,也就是说,如果你在一个不是main的函数中,finish
命令会导致GDB恢复执行,直到恰好在函数返回之后为止。
以下使用代码1调试:
(gdb) b display 
Breakpoint 1 at 0x4004ff: file gdb.c, line 5.
(gdb) r
Starting program: /home/yanwenjie/ctest/a.out 

Breakpoint 1, display (i=1) at gdb.c:5
5        i = i + 1;
(gdb) finish
Run till exit from #0  display (i=1) at gdb.c:5
i = 2
main () at gdb.c:13
13        return 0;
(gdb) 

6.2.until
until(简写为u)会执行循环的其余部分,让GDB在循环后面的第一行代码处暂停,条件时循环中不能有断点。
使用代码2进行调试:
(gdb) b main
(gdb) r
Breakpoint 1, main () at gdb.c:5
5        int m = 1;
(gdb) n
7        for (i=0; i<10; i++) {
(gdb) 
8 printf("%d",i);
(gdb) 
7        for (i=0; i<10; i++) {
(gdb) until
10        return 0;
(gdb) n
11 }

7.条件断点

条件断点告诉调试器只有当符合某种条件时才在断点处停止,比如当变量具有某个特定的感兴趣的值时。
设置条件断点的语法为:
break break-args if (condition)
条件终端极其灵活,主要包含如下运算符:
1.相等、逻辑或不相等运算符(<、<=、==、!=、>、>= 、&&、||等),例如:
break 180 if string == NULL && i < 0
2.按位和移位运算符(&、|、^、>>、<<等),例如:
break test.c:34 if(x & y) == 1
3.算术运算符(+、-、x、/、%),例如:
break myfunc if i % (j+3) != 0
4.你自己的函数,只要它们被链接到程序中,例如:
break test.c:myfunc if ! check_variable(i)
5.库函数,只要该库被链接到代码中,例如:
break 44 if stelen(mystring) == 0

8.断点命令列表

当gdb遇到断点时,几乎总是要查看某个变量,如果反复遇到同一个断点,将反复查看相同的变量,让GDB在每次到达某个断点时
自动执行一组命令,从而完成一个过程,这就是断点命令列表可以完成的事情。
使用commands命令设置命令列表。
commands命令设置命令列表。
commands breakpoint-num
.....
commands
....
end
其中breakpoint-number是要将命令添加到其上的断点的标识符。commands是用新行分割任何有效gdb命令列表。逐条输入命令,
然后键入end表示输入命令完毕。从那以后,每当gdb在这个断点处中断时,它会执行你输入的任何命令。
使用代码2进行调试:
(gdb) i b
Num     Type           Disp Enb Address            What
1       breakpoint     keep y   0x000000000040050c in main at gdb.c:8
(gdb) commands 1
Type commands for breakpoint(s) 1, one per line.
End with a line saying just "end".
>printf "i is %d\n",i
>end
(gdb) r
Breakpoint 1, main () at gdb.c:8
8 printf("%d",i);
i is 0
(gdb) c
Continuing.


Breakpoint 1, main () at gdb.c:8
8 printf("%d",i);
i is 1

9.监视点

监视点是一种特殊类型的断点,它类似与正常断点,是要求GDB暂停程序执行的指令。监视点指示gdb每当某个表达式改变了
值就暂停执行。当变量i存在作用域中时,使用如下方法设置监视点:
watch i
它会使得每次i改变时gdb暂停。
使用如下代码调试:
#include <stdio.h>int main(void)
{int m = 1;int i;for (i=0; i<10; i++) {if (i%2 ==0) {m++;}}return 0;
}
调试结果如下:
(gdb) b main
Breakpoint 1 at 0x4004b8: file gdb.c, line 5.
(gdb) r
Breakpoint 1, main () at gdb.c:5
5        int m = 1;
(gdb) watch m
Hardware watchpoint 2: m
(gdb) p m
$1 = 0
(gdb) c
Continuing.
Hardware watchpoint 2: m


Old value = 0  
New value = 2
(好像对m的赋值没有作为监控点而停住)
main () at gdb.c:7
7        for (i=0; i<10; i++) {
(gdb) p m
$2 = 2

这篇关于《软件调试的艺术》笔记--停下来环顾程序的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/904109

相关文章

IDEA如何实现远程断点调试jar包

《IDEA如何实现远程断点调试jar包》:本文主要介绍IDEA如何实现远程断点调试jar包的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录问题步骤总结问题以jar包的形式运行Spring Boot项目时报错,但是在IDEA开发环境javascript下编译

Python程序打包exe,单文件和多文件方式

《Python程序打包exe,单文件和多文件方式》:本文主要介绍Python程序打包exe,单文件和多文件方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录python 脚本打成exe文件安装Pyinstaller准备一个ico图标打包方式一(适用于文件较少的程

如何确定哪些软件是Mac系统自带的? Mac系统内置应用查看技巧

《如何确定哪些软件是Mac系统自带的?Mac系统内置应用查看技巧》如何确定哪些软件是Mac系统自带的?mac系统中有很多自带的应用,想要看看哪些是系统自带,该怎么查看呢?下面我们就来看看Mac系统内... 在MAC电脑上,可以使用以下方法来确定哪些软件是系统自带的:1.应用程序文件夹打开应用程序文件夹

Python程序的文件头部声明小结

《Python程序的文件头部声明小结》在Python文件的顶部声明编码通常是必须的,尤其是在处理非ASCII字符时,下面就来介绍一下两种头部文件声明,具有一定的参考价值,感兴趣的可以了解一下... 目录一、# coding=utf-8二、#!/usr/bin/env python三、运行Python程序四、

无法启动此程序因为计算机丢失api-ms-win-core-path-l1-1-0.dll修复方案

《无法启动此程序因为计算机丢失api-ms-win-core-path-l1-1-0.dll修复方案》:本文主要介绍了无法启动此程序,详细内容请阅读本文,希望能对你有所帮助... 在计算机使用过程中,我们经常会遇到一些错误提示,其中之一就是"api-ms-win-core-path-l1-1-0.dll丢失

SpringBoot后端实现小程序微信登录功能实现

《SpringBoot后端实现小程序微信登录功能实现》微信小程序登录是开发者通过微信提供的身份验证机制,获取用户唯一标识(openid)和会话密钥(session_key)的过程,这篇文章给大家介绍S... 目录SpringBoot实现微信小程序登录简介SpringBoot后端实现微信登录SpringBoo

uniapp小程序中实现无缝衔接滚动效果代码示例

《uniapp小程序中实现无缝衔接滚动效果代码示例》:本文主要介绍uniapp小程序中实现无缝衔接滚动效果的相关资料,该方法可以实现滚动内容中字的不同的颜色更改,并且可以根据需要进行艺术化更改和自... 组件滚动通知只能实现简单的滚动效果,不能实现滚动内容中的字进行不同颜色的更改,下面实现一个无缝衔接的滚动

Python MCPInspector调试思路详解

《PythonMCPInspector调试思路详解》:本文主要介绍PythonMCPInspector调试思路详解,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋... 目录python-MCPInspector调试1-核心知识点2-思路整理1-核心思路2-核心代码3-参考网址

Java使用WebView实现桌面程序的技术指南

《Java使用WebView实现桌面程序的技术指南》在现代软件开发中,许多应用需要在桌面程序中嵌入Web页面,例如,你可能需要在Java桌面应用中嵌入一部分Web前端,或者加载一个HTML5界面以增强... 目录1、简述2、WebView 特点3、搭建 WebView 示例3.1 添加 JavaFX 依赖3

防止SpringBoot程序崩溃的几种方式汇总

《防止SpringBoot程序崩溃的几种方式汇总》本文总结了8种防止SpringBoot程序崩溃的方法,包括全局异常处理、try-catch、断路器、资源限制、监控、优雅停机、健康检查和数据库连接池配... 目录1. 全局异常处理2. 使用 try-catch 捕获异常3. 使用断路器4. 设置最大内存和线