news 2026/3/1 11:06:23

【Linux 基础知识系列:第二百零四篇】使用 gdb 调试 C 程序入门

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
【Linux 基础知识系列:第二百零四篇】使用 gdb 调试 C 程序入门

一、简介:为什么必须掌握 gdb?

  • C 语言贴近硬件,指针越界、段错误、内存泄漏悄无声息,编译期无法发现。

  • printf 打桩效率低、破坏线程时序、难以观察复杂结构体。

  • gdb(GNU Debugger)是 Linux 下事实标准调试器,支持:

    • 断点/单步/回退执行

    • 查看变量、内存、寄存器

    • 多线程、Core 文件、远程调试

  • 应用场景

    • 嵌入式开发(ARM/Linux)

    • 高性能服务器(Nginx、Redis)

    • 算法竞赛现场(ACM/OI 允许携带 gdb 脚本)

学会 gdb = 给 C 程序装“外科手术灯”,定位问题从“小时”级缩短到“分钟”级。


二、核心概念:5 个单词先记牢

名词一句话说明本文出现形式
断点(breakpoint)让程序暂停的地址/行号b main
单步(step/next)逐行执行,step 进入函数,next 不进入s/n
回溯(backtrace)查看调用栈,秒级定位段错误bt
Core 文件程序崩溃时的内存快照,可事后调试gdb a.out core
TUIgdb 的文本图形界面,代码窗口+命令窗口ctrl+x a

三、环境准备:3 行命令搞定

  1. 操作系统
    Ubuntu 20.04+ / CentOS 8+ / WSL2 均可(内核无要求)。

  2. 安装 gdb

    sudo apt update && sudo apt install -y gdb gcc # Debian/Ubuntu sudo dnf install -y gdb gcc # CentOS/RHEL
  3. 确认版本

    gdb --version | head -1 # ≥ 8.0 支持语法高亮
  4. 实验目录

    mkdir -p ~/gdb-lab && cd ~/gdb-lab

四、实际案例与步骤:从 10 行小程序到 Core dump

每个示例均可直接复制,保存后chmod +x run.sh && ./run.sh一键跑通。


4.1 编译就要带调试信息:-g -O0

# file: hello.c #include <stdio.h> int main(){ int a = 1; printf("a=%d\n", a); return 0; } # 编译命令(记住模板) gcc -g -O0 hello.c -o hello

要点

  • -g生成调试符号;-O0关闭优化,防止变量被优化消失。


4.2 快速体验:启动→断点→打印→继续

gdb hello -ex "b main" -ex "r" -ex "p a" -ex "c" -ex "q"

一行命令拆解

片段作用
gdb hello加载可执行文件
-ex "b main"设置断点
-ex "r"run
-ex "p a"print 变量 a
-ex "c"continue
-ex "q"退出

输出

Breakpoint 1, main () at hello.c:4 4 int a = 1; (gdb) p a $1 = 1

恭喜,你已完成人生第一次 gdb 调试


4.3 交互式调试:单步 + 查看源码

gdb hello (gdb) b main (gdb) r (gdb) n # next,不进入 printf (gdb) l # list,显示源码 (gdb) p a (gdb) s # step,会进入 printf 库函数(若想看可进入) (gdb) q

快捷键
n/s/c/q养成肌肉记忆。


4.4 段错误经典案例:空指针解引用

// file: crash.c #include <stdio.h> int main(){ int *p = NULL; *p = 42; // 段错误 return 0; }

编译 & 运行:

gcc -g -O0 crash.c -o crash ./crash # 屏幕输出:Segmentation fault (core dumped)

调试步骤

# 1. 直接 gdb 跑 gdb crash -ex r # 2. 自动停在 SIGSEGV (gdb) bt # 输出 #0 main () at crash.c:5 # 3. 看变量 (gdb) p p $1 = (int *) 0x0

结论p是空指针,*p = 42导致段错误,3 秒定位


4.5 Core 文件事后调试:生产环境必用

  1. 打开 core 开关

    ulimit -c unlimited echo "core.%e.%p" | sudo tee /proc/sys/kernel/core_pattern
  2. 重新运行 crash
    目录下生成core.crash.12345

  3. 事后调试

    gdb crash core.crash.12345 -ex bt -ex q

    无需重新运行程序,现场保留


4.6 多线程调试:生产者-消费者示例

// file: pc.c(片段,完整代码见文末 GitHub) pthread_mutex_t mtx; pthread_cond_t cv; ... void* producer(void* arg){ pthread_mutex_lock(&mtx); while (count == MAX) pthread_cond_wait(&cv, &mtx); buffer[in] = rand(); in = (in+1)%MAX; count++; pthread_mutex_unlock(&mtx); }

编译:

gcc -g -O0 pc.c -o pc -pthread

调试:

gdb pc (gdb) b producer (gdb) r (gdb) info threads # 查看线程 (gdb) thread 2 # 切换线程 (gdb) bt

技巧
set print thread-events off减少线程切换提示噪音。


4.7 TUI 文本界面:告别黑白屏

gdb pc -tui # 或内部:ctrl+x a

界面拆分:

  • 上半部:源码窗口

  • 下半部:gdb 命令窗口
    快捷键ctrl+x 2打开汇编/寄存器窗口,炫酷又实用


4.8 VS Code 图形调试(bonus)

  1. 安装插件C/C++(ms-vscode.cpptools)

  2. F5→ 选择gdb→ 自动生成.vscode/launch.json

  3. 关键片段(可复制)

    { "name": "gdb launch", "type": "cppdbg", "request": "launch", "program": "${workspaceFolder}/crash", "miDebuggerPath": "/usr/bin/gdb", "stopAtEntry": false, "cwd": "${workspaceFolder}", "externalConsole": false }
  4. 打断点、监视变量、调用栈图形化,新手友好


五、常见问题与解答(FAQ)

问题现象解决
No symbol table is loaded.忘了-g重新gcc -g -O0编译
ptrace: Operation not permitted.容器/Seccomp 限制--cap-add=SYS_PTRACE
打印 STL 容器乱码std::vector显示 `{...}``安装gdb-pretty-printer或升级 gdb≥9
单步进入汇编s后看到__libc_start_main汇编n直到源码行,或set step-mode off
Core 文件太大磁盘爆满限制大小ulimit -c 102400或使用systemd-coredump

六、实践建议与最佳实践

  1. 编译脚本模板(保存为build.sh

    #!/bin/bash set -e gcc -g -O0 -Wall -Wextra "$1.c" -o "$1" "${@:2}"

    用法:./build.sh crash -pthread

  2. .gdbinit 个人配置

    set print pretty on set print array on set confirm off set history save on
  3. 自动化小脚本:一键 backtrace

    # usage: gdb-bt <prog> <core> gdb -batch -ex bt "$1" "$2"
  4. 远程调试(嵌入式)
    目标板:gdbserver :1234 ./app
    本地:gdb app -ex "target remote 192.168.1.100:1234"

  5. 生产环境

    • 永远保留未裁剪二进制(-g)到调试仓库。

    • 使用strip --only-keep-debug app app.debug分离符号,减小发布体积。


七、总结:一张脑图带走全部要点

gdb 调试路线图 ├─ 编译:gcc -g -O0 ├─ 启动:gdb hello → b main → r ├─ 单步:n / s / finish ├─ 查看:p var / bt / info locals ├─ 高级:core / TUI / VS Code / gdbserver └─ 习惯:.gdbinit + 自动化脚本

掌握 gdb,你就拥有了:

  • 段错误3 分钟定位能力

  • 多线程死锁可视化调用栈

  • Core dump事后复盘,无需现场重新运行

  • 远程/图形/自动化三套环境无缝切换

立刻打开终端,复制本文命令跑一遍,把build.sh.gdbinit加入你的 Git 仓库——从此告别 printf 打桩,调试效率翻倍!祝你玩的开心,代码无 bug。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/3/1 5:16:49

速藏!AI大模型学习指南:从新手入门到实战通关

当AI大模型能自动生成技术方案、辅助调试代码&#xff0c;甚至独立开发简单应用时&#xff0c;它已不再是技术圈的“奢侈品”&#xff0c;而是每个程序员、IT从业者的“必备技能包”。无论是想转行AI的职场人、刚接触编程的小白&#xff0c;还是寻求技术突破的开发者&#xff0…

作者头像 李华
网站建设 2026/2/28 4:14:21

攻防世界——心仪的公司

拿到数据包先尝试用关键字搜索flag 只找到一个非正常格式的flag 尝试依据数据包名称来搜索关键字&#xff08;webshell&#xff09; 发现都指向webshell.jpg这个文件 并且由第二张图的路径和文件名可知是用web中的http协议进行传输 精准查询webshell文件 http.file_data co…

作者头像 李华
网站建设 2026/2/28 1:59:36

种子扩散预览:兼具快速推理和高性能的下一代代码生成模型

概述 本研究提出了基于离散状态扩散&#xff08;DSD&#xff09;的快速推理模型–种子扩散预览&#xff08;Seed Diffusion Preview&#xff09;&#xff0c;作为大规模语言建模的一种新方法。 传统的自回归&#xff08;AR&#xff09;模型是按顺序生成标记的&#xff0c;这限制…

作者头像 李华
网站建设 2026/2/28 6:20:12

多路召回(Multi-Route Retrieval)

核心原理与技术架构多路召回通过并行执行多个检索策略&#xff0c;覆盖不同语义维度&#xff08;关键词、语义、多模态等&#xff09;&#xff0c;再通过融合算法&#xff08;如 RRF、加权排名&#xff09;生成最终结果&#xff0c;解决单一检索的 “漏检” 和 “错配” 问题。…

作者头像 李华