1.准备工作
使用gdb -v
可以查看自己linux环境是否安装了gdb,如果安装了,那么会显示如下效果:
如果没有安装,那么可以按照下面的步骤进行安装(此处针对的是CentOS 为例),一行代码就够了:
sudo yum -y install gdb
然后要准备好我们要调试的.c或者.cpp文件,我这里先简单的用了一个.c文件进行演示。首先使用命令gcc -g fork2.c
说明: 加个-g 是为了gdb 用,因为直接生成的可执行文件不包括行号,无法进行gdb调试
执行完之后我们使用ls命令查看当前目录,会发现有a.out文件,这时我们就可以进入第二阶段的调试工作了。
2.启动调试
输入gdb a.out
命令,会进入这一阶段,然后我们需要打断点。
在(gdb)的后面输入b main,b表示的是打断点,表示我们在main函数入口设置了一个断点。也可以用b 2 或者b 10表示在第2行或者第10行进行打断点。
之后,我们在(gdb)后面每输入一次n,代码就会往下执行一步。
下图为对应的源代码,可以进行对比观察。
在这个阶段,我将演示对mutex.cpp文件进行gdb调试。下面给出了一个有关多线程中互斥锁小测试的mutex.cpp代码
#include<stdio.h>
#include<stdlib.h>
#include<pthread.h>
#include<unistd.h>
int ticket_sum = 10;
//创建互斥锁
pthread_mutex_t myMutex = PTHREAD_MUTEX_INITIALIZER;
//模拟售票员卖票
void *sell_ticket(void *arg) {
//输出当前执行函数的线程 ID
printf("当前线程ID:%u\n", pthread_self());
int i;
int islock = 0;
for (i = 0; i < 10; i++)
{
//当前线程“加锁”
islock = pthread_mutex_lock(&myMutex);
//如果“加锁”成功,执行如下代码
if (islock == 0) {
//如果票数 >0 ,开始卖票
if (ticket_sum > 0)
{
sleep(1);
printf("%u 卖第 %d 张票\n", pthread_self(), 10 - ticket_sum + 1);
ticket_sum--;
}
//当前线程模拟完卖票过程,执行“解锁”操作
pthread_mutex_unlock(&myMutex);
}
}
return 0;
}
int main() {
int flag;
int i;
void *ans;
//创建 4 个线程,模拟 4 个售票员
pthread_t tids[4];
for (i = 0; i < 4; i++)
{
flag = pthread_create(&tids[i], NULL, &sell_ticket, NULL);
if (flag != 0) {
printf("线程创建失败!");
return 0;
}
}
sleep(10); //等待 4 个线程执行完成
for (i = 0; i < 4; i++)
{
//阻塞主线程,确认 4 个线程执行完成
flag = pthread_join(tids[i], &ans);
if (flag != 0) {
printf("tid=%d 等待失败!", tids[i]);
return 0;
}
}
return 0;
}
在linux中,如果用到了线程函数,那么在使用gcc编译的时候需要额外加上-lpthread命令。
由于我们要对其进行gdb调试,因此在编译的时候需要输入以下命令:
gcc mutex.cpp -o mutex.exe -g -lpthread
效果如下:
现在我们对其进行gdb调试,这次教点不一样的。
-
首先输入
gdb mutex.exe
对其进行调试 -
然后输入
l
,可以显示带行号的源代码(默认情况下,l
选项只显示 10 行源代码,如果查看后续代码,按Enter
回车即可)
效果图如下: -
从上面给出的源码可以看到,我们可以在41行打断点,然后查看当前i的值。
-
使用
b 41
,按回车键,然后再输入r
,再按回车键,发现此时程序就会执行到断点处
-
按n是一步一步运行,这个之前讲到过,按r是会运行到打断点的地方,所以要提前打好断点,免得直接步出了。
-
最后,按q是直接退出调试。
如果觉得对你有所帮助的话,希望能点赞收藏一波,您的鼓励就是对我最大的支持,谢谢!