前言

使用gdb来调试程序应该算是Linux开发的必备技能吧,今天来看看如何使用gdb调试段错误(Segmentation fault)。

如何调试段错误

首先来看一下如下例子:

#include <stdio.h>
#include <stdlib.h>

int main(int argc, char **argv)
{
char *buf;

buf = malloc(1<<31);

fgets(buf, 1024, stdin);
printf("%s\n", buf);

return 1;
}

然后编译执行

gdb调试段错误_空指针

执行时出现了段错误(Segmentation fault)。那接下来我们来使用gdb排查问题在哪?

我们重新编译程序,但是编译时要加上-g,要使用gdb调试时,编译要加上该选项。

gdb调试段错误_#include_02

接下来就是使用gdb进行调试了!

执行gdb test

gdb调试段错误_gdb调试_03

然后使用run开始执行程序

gdb调试段错误_linux_04

我们收到了来自操作系统的SIGSEGV信号,这就意味着我们试图去访问一段非法的内存。

我们使用backtrace看一下堆栈情况:

gdb调试段错误_gdb调试_05

我们主要看我们程序中的,提示我们崩溃在第10行!就是

fgets(buf, 1024, stdin);

为什么会崩溃在fgets呢,肯定是我们给的参数有问题!我们使用print打印一下buf的值。

gdb调试段错误_段错误_06

呵呵!空指针........那就是malloc没成功了!我们打断点来调试。先使用kill停止

gdb调试段错误_linux_07

接下来在第8行打断点

gdb调试段错误_段错误_08

看清楚,打断点的方式,是在C文件上:

break test.c:8

然后run就停在了第8行,我们打印一下buf值

gdb调试段错误_#include_09

我们发现执行malloc前后都是0x0, 当然有时候malloc前可能是一个乱值。

这已经可以说明malloc没有成功了!其实主要就是我们分配的空间太大了,分配了4G左右的空间。改为1024就OK了!


总结

上面主要还是跟大家讲一些基本的排查方法,gdb还有挺多功能的,大家可以去查阅资料。gdb在应用开发还是比较常用的,大家还是要掌握!