GDB断点学习(断点设置)

GDB是调试利器,在某些场景下,不确定是否好用,今天特意测试了在虚函数,内联函数,函数模板的使用

#include <iostream>
using namespace std;

class Base
{
public:
	virtual void test(){
		cout << "i am base" << endl;
	}
};

class Derived: public Base
{
private:
	int a_;
public:
	inline void testInline(int b)
	{
		a_++;
		a_ += b;
		a_ += 7;
	}

	void callInline()
	{
		int b = 111;
		testInline(b);
	}
	virtual void test() override {
		cout << "i am Derived" << endl;
	}
};

template<typename T>
T MaxElement(T a, T b, T c)
{
	T tmp = a;
	if (tmp < b) {
		tmp = b;
	}
	if (tmp < c) {
		tmp = c;
	}
	cout << "max element: " << tmp << endl;
	return tmp;
}


int main()
{
	double maxDouble = MaxElement(1.2, 3.0, -3.2);
	float maxFloat = MaxElement(121.2f, 23.0f, -31.2f);
	int maxInt = MaxElement(1,2,3);
	Derived d;
	d.test();
	d.callInline();
	
	return 0;
}
aaa@ubuntu:~/share$ g++ -std=c++11 -g main.cpp -o main
aaa@ubuntu:~/share$ gdb main
GNU gdb (Ubuntu 7.11.1-0ubuntu1~16.5) 7.11.1
(gdb) b MaxElement  #对于模板函数,这样是不行滴
Function "MaxElement" not defined.
Make breakpoint pending on future shared library load? (y or [n]) y
Breakpoint 1 (MaxElement) pending.
(gdb) b MaxElement<int> #需要这样子
Breakpoint 2 at 0x400db4: file main.cpp, line 37.
(gdb) b MaxElement<double> #需要这样子
Breakpoint 3 at 0x400cae: file main.cpp, line 37.
(gdb) b MaxElement<float>  #需要这样子
Breakpoint 4 at 0x400d36: file main.cpp, line 37.
(gdb) b MaxElement<short> #因为没有short的调用,也不行
Function "MaxElement<short>" not defined.
Make breakpoint pending on future shared library load? (y or [n]) n
(gdb) b Derived::testInline  #inline函数断点
Breakpoint 5 at 0x400bcb: file main.cpp, line 19.
(gdb) b Derived::test #虚函数断点
Breakpoint 6 at 0x400c38: file main.cpp, line 30.
(gdb) r
Starting program: /home/cherish/share/main 

Breakpoint 3, MaxElement<double> (a=1.2, b=3, c=-3.2000000000000002) at main.cpp:37
37              T tmp = a;
(gdb) r
The program being debugged has been started already.
Start it from the beginning? (y or n) n
Program not restarted.
(gdb) c
Continuing.
max element: 3

Breakpoint 4, MaxElement<float> (a=121.199997, b=23, c=-31.2000008) at main.cpp:37
37              T tmp = a;
(gdb) c
Continuing.
max element: 121.2

Breakpoint 2, MaxElement<int> (a=1, b=2, c=3) at main.cpp:37
37              T tmp = a;
(gdb) c
Continuing.
max element: 3

Breakpoint 6, Derived::test (this=0x7fffffffe480) at main.cpp:30
30                      cout << "i am Derived" << endl;
(gdb) c
Continuing.
i am Derived

Breakpoint 5, Derived::testInline (this=0x7fffffffe480, b=111) at main.cpp:19
19                      a_++;
(gdb) c
Continuing.
[Inferior 1 (process 4921) exited normally]
(gdb) c

对于模板函数,还有一个黑科技,一次设置所有类型的断点(rbreak)。这里看到,通过这个命令可以一次性把MaxElement实例的函数都设置了断点。 

(gdb) rbreak MaxElement<.*>
Breakpoint 7 at 0x400cae: file main.cpp, line 37.
double MaxElement<double>(double, double, double);
Note: breakpoint 4 also set at pc 0x400d36.
Breakpoint 8 at 0x400d36: file main.cpp, line 37.
float MaxElement<float>(float, float, float);
Breakpoint 9 at 0x400db4: file main.cpp, line 37.
int MaxElement<int>(int, int, int);

在函数模板里,如果我们仅仅对某个类型的实例设置断点,那么其他类型的实例在允许时是不会暂停的。如下

aaa@ubuntu:~/share$ gdb main
GNU gdb (Ubuntu 7.11.1-0ubuntu1~16.5) 7.11.1
(gdb) info b
No breakpoints or watchpoints.

(gdb) b  MaxElement<float>
Breakpoint 1 at 0x400d36: file main.cpp, line 37.
(gdb) info b
Num     Type           Disp Enb Address            What
1       breakpoint     keep y   0x0000000000400d36 in MaxElement<float>(float, float, float) at main.cpp:37

(gdb) r
Starting program: /home/cherish/share/main 
max element: 3

Breakpoint 1, MaxElement<float> (a=121.199997, b=23, c=-31.2000008) at main.cpp:37
37              T tmp = a;

(gdb) c
Continuing.
max element: 121.2
max element: 3
i am Derived
[Inferior 1 (process 4937) exited normally]

 

最后如果去除了符号表,那么是如设置断点的,如下:

aaa@ubuntu:~/share$ g++ -std=gnu++11 main.cpp -o main_with_sym 
aaa@ubuntu:~/share$ g++ -std=gnu++11 -s main.cpp -o main_without_sym  #去除符号表
aaa@ubuntu:~/share$ ls -l main*
-rwxrw-rw- 1 aaa aaa   816 Mar 12 08:06 main.cpp
-rwxrwxr-x 1 aaa aaa 10488 Mar 12 08:54 main_without_sym
-rwxrwxr-x 1 aaa aaa 14448 Mar 12 08:54 main_with_sym
aaa@ubuntu:~/share$ objdump -t main_without_sym 

main_without_sym:     file format elf64-x86-64

SYMBOL TABLE:
no symbols


aaa@ubuntu:~/share$ objdump -t main_with_sym    

main_with_sym:     file format elf64-x86-64

SYMBOL TABLE:
0000000000400238 l    d  .interp        0000000000000000              .interp
0000000000400254 l    d  .note.ABI-tag  0000000000000000              .note.ABI-tag
......
0000000000400bc0  w    F .text  0000000000000043              _ZN7Derived10testInlineEi
0000000000400e80 g     F .text  0000000000000002              __libc_csu_fini
0000000000400c2c  w    F .text  000000000000002b              _ZN7Derived4testEv
0000000000400980 g     F .text  000000000000002a              _start
0000000000000000  w      *UND*  0000000000000000              __gmon_start__
0000000000000000  w      *UND*  0000000000000000              _Jv_RegisterClasses
0000000000400c70  w    F .text  0000000000000027              _ZN7DerivedC1Ev
0000000000400e84 g     F .fini  0000000000000000              _fini
0000000000000000       F *UND*  0000000000000000              _ZNSt8ios_base4InitC1Ev@@GLIBCXX_3.4
0000000000000000       F *UND*  0000000000000000              __libc_start_main@@GLIBC_2.2.5
0000000000400d1f  w    F .text  0000000000000084              _Z10MaxElementIfET_S0_S0_S0_
0000000000400f10  w    O .rodata        0000000000000018              _ZTI7Derived
0000000000000000       F *UND*  0000000000000000              __cxa_atexit@@GLIBC_2.2.5
0000000000400910       F *UND*  0000000000000000              _ZNSt8ios_base4InitD1Ev@@GLIBCXX_3.4
0000000000000000  w      *UND*  0000000000000000              _ITM_deregisterTMCloneTable
0000000000000000       F *UND*  0000000000000000              _ZStlsISt11char_traitsIcEERSt13basic_ostreamIcT_ES5_PKc@@GLIBCXX_3.4
0000000000400e90 g     O .rodata        0000000000000004              _IO_stdin_used
0000000000400c58  w    F .text  0000000000000017              _ZN4BaseC1Ev
0000000000000000  w      *UND*  0000000000000000              _ITM_registerTMCloneTable
0000000000602070 g       .data  0000000000000000              __data_start
0000000000400ee0  w    O .rodata        0000000000000018              _ZTV7Derived
0000000000400da3  w    F .text  0000000000000064              _Z10MaxElementIiET_S0_S0_S0_
0000000000602080  w    O .bss   0000000000000058              _ZTVN10__cxxabiv117__class_type_infoE@@CXXABI_1.3
0000000000400c97  w    F .text  0000000000000088              _Z10MaxElementIdET_S0_S0_S0_
......
00000000006021f0  w    O .bss   0000000000000058              _ZTVN10__cxxabiv120__si_class_type_infoE@@CXXABI_1.3
0000000000400c04  w    F .text  0000000000000027              _ZN7Derived10callInlineEv
0000000000000000       F *UND*  0000000000000000              __stack_chk_fail@@GLIBC_2.4
0000000000000000       F *UND*  0000000000000000              _ZNSolsEf@@GLIBCXX_3.4
0000000000400b94  w    F .text  000000000000002b              _ZN4Base4testEv
0000000000602250 g       .bss   0000000000000000              _end
0000000000000000       F *UND*  0000000000000000              _ZNSolsEPFRSoS_E@@GLIBCXX_3.4
0000000000400c58  w    F .text  0000000000000017              _ZN4BaseC2Ev
0000000000400960       F *UND*  0000000000000000              _ZSt4endlIcSt11char_traitsIcEERSt13basic_ostreamIT_T0_ES6_@@GLIBCXX_3.4
0000000000400ef8  w    O .rodata        0000000000000018              _ZTV4Base
0000000000602080 g       .data  0000000000000000              _edata
0000000000400a76 g     F .text  00000000000000ca              main
0000000000400888 g     F .init  0000000000000000              _init


aaa@ubuntu:~/share$ gdb ./main
main.cpp          main_without_sym  main_with_sym     
aaa@ubuntu:~/share$ gdb ./main_without_sym 
(gdb) b main
Function "main" not defined. #找不到函数
Make breakpoint pending on future shared library load? (y or [n]) y
Breakpoint 1 (main) pending.
(gdb) r  #直接执行完毕
Starting program: /home/aaa/share/main_without_sym 
max element: 3
max element: 121.2
max element: 3
i am Derived
[Inferior 1 (process 5180) exited normally]
(gdb) q
aaa@ubuntu:~/share$ gdb ./main_with_sym      

(gdb) b main
Breakpoint 1 at 0x400a7a  #找到函数
(gdb) r
Starting program: /home/aaa/share/main_with_sym  
Breakpoint 1, 0x0000000000400a7a in main () #停在断点
(gdb) c
Continuing.
max element: 3
max element: 121.2
max element: 3
i am Derived
[Inferior 1 (process 5186) exited normally]
(gdb) 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在Linux中使用gdb调试程序时,可以通过设置断点来暂停程序的执行,以便进行调试。设置断点的方法如下: 1. 在终端中打开gdb调试器,输入要调试的程序的路径和名称,例如:gdb /path/to/program 2. 进入gdb调试器后,可以使用b命令设置断点。例如,要在程序的第10行设置断点,可以输入:b 10 3. 如果要在某个函数中设置断点,可以使用b命令加上函数名。例如,要在函数foo()中设置断点,可以输入:b foo 4. 如果要在某个文件中设置断点,可以使用b命令加上文件名和行号。例如,要在文件test.c的第20行设置断点,可以输入:b test.c:20 5. 设置断点后,可以使用r命令运行程序。程序执行到断点处时,会自动暂停执行。 6. 在程序暂停执行时,可以使用s命令单步执行程序,或使用c命令继续执行程序直到下一个断点。 7. 如果要删除断点,可以使用d命令。例如,要删除第10行的断点,可以输入:d 10 以上就是在Linux中使用gdb调试器设置断点的方法。 ### 回答2: GDB是一个功能强大的调试工具,可以帮助我们在Linux环境下调试程序并诊断问题。为了更好地使用GDB进行调试,我们需要了解如何设置断点。下面将详细介绍如何在Linux中使用GDB设置断点。 首先,我们需要编写一个需要调试的程序,并使用-g选项编译程序。例如,我们编写一个名为test的C程序并使用以下命令编译它: gcc -g test.c -o test 在编译过程中,-g选项会在程序中生成用于调试的符号表。这些符号表包含程序中的变量和函数名称等信息,从而使GDB能够准确地显示代码和调试信息。 接下来,我们需要使用GDB打开编译过的程序: gdb test 当GDB启动时,我们可以使用run命令运行程序。如果程序需要参数,则可以在run命令后添加参数。例如,如果我们的程序需要一个文件名作为参数,则可以使用以下命令运行程序: run filename 现在,我们可以在程序中设置断点设置断点的方法有许多种,以下是几种常用的方法: 1.使用break命令设置断点break命令后可以跟一个文件名和行号,表示在该文件的该行设置断点。例如,要在test.c的第10行设置断点,可以使用以下命令: break test.c:10 如果要设置在一个函数中设置断点,可以在break命令后面跟上函数名。例如,要在main函数中设置断点,可以使用以下命令: break main 2.使用tbreak命令设置临时断点。tbreak命令与break命令类似,但是设置断点只在第一次触发后失效。例如,要在test.c的第20行设置临时断点,可以使用以下命令: tbreak test.c:20 3.在程序运行时使用ctrl+c暂停程序,然后使用break命令设置断点。这种方法适用于在程序执行过程中才能确定需要设置断点的情况。例如,要在程序运行到函数foo()时设置断点,可以在程序运行时使用ctrl+c暂停程序,然后使用以下命令设置断点break foo 设置断点后,我们可以使用continue命令让程序继续运行,直到程序执行到设置断点处停止。例如,要让程序执行到设置断点处停止,可以使用以下命令: continue 设置断点是一个非常重要的调试技巧,它可以帮助我们在程序执行过程中找到问题。在使用GDB进行调试时,我们需要充分利用这个功能,不断迭代代码,并在调试过程中修复问题。 ### 回答3: GDB是Linux下最常用的调试工具之一,它的用法比较复杂,但是只要掌握了基本的调试技巧,就能够快速定位到代码的错误并进行修复。 设置断点GDB调试的基本操作之一,通过设置断点,我们可以在程序执行的某个地方停止,并且查看此时的程序状态。下面我们来详细介绍在Linux GDB中如何设置断点。 1. 启动GDB。 在终端中输入"gdb"启动GDB。 2. 加载可执行文件。 可以使用GDB的"file"命令来加载可执行文件,在终端中输入: ``` (gdb) file 可执行文件路径 ``` 此时,GDB就会加载相应的可执行文件。 3. 设置断点设置断点可以使用GDB的"break"命令,可以在指定行数或者函数处设置断点。在终端中输入: - 在指定行数上设置断点 ``` (gdb) break 行号 ``` - 在指定函数处设置断点 ``` (gdb) break 函数名 ``` 设置断点后,可以通过输入"info break"命令查看已经设置断点信息。 4. 运行程序。 设置断点之后,可以通过输入"run"命令来运行被调试的程序,程序会在第一个断点处停止。 5. 调试程序。 程序停在断点处之后,可以使用GDB的"next"、"step"、"print"等命令来进行调试,这些命令的作用分别是: - next:单步执行代码,不进入函数调用。 - step:单步执行代码,进入函数调用。 - print:打印变量值。 调试过程中,可以使用"continue"命令让程序继续执行,直到下一个断点处停止。 6. 删除断点。 如果要删除设置断点,可以使用GDB的"delete"命令,可以在终端中输入: - 删除指定行数处的断点 ``` (gdb) delete 行号 ``` - 删除所有断点 ``` (gdb) delete ``` 以上就是在Linux GDB设置断点的基本步骤和命令,通过学习和练习,相信大家都可以掌握GDB的基本调试技巧,从而提高程序调试的效率。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值