C++程序运行需经预处理、编译、汇编、链接、加载、执行六阶段;预处理做文本替换,编译检查语法并生成汇编,汇编产出含符号表的目标文件,链接解析跨文件引用生成可执行文件。

C++程序从写完代码到真正运行,要经过编译、链接、加载、执行四个关键阶段。每个阶段都可能出错,理解流程能帮你快速定位问题,比如“找不到函数”是链接阶段的问题,“段错误”往往发生在运行时。
预处理:处理 #include、#define 和条件编译
源文件(.cpp)先交给预处理器。它不关心语法是否正确,只做文本替换:把 #include iostream> 展开成 iostream 的实际内容,把 #define PI 3.14 替换成数字,删掉被 #ifdef 排除的代码块。这一步输出的是一个“.ii”文件(或直接传给编译器),体积通常大很多。
编译:把 C++ 代码翻译成汇编指令
编译器(如 g++)对预处理后的代码做词法、语法、语义分析。检查变量是否声明、函数调用是否匹配、模板能否实例化等。通过后生成对应平台的汇编代码(.s 文件)。注意:此时每个 .cpp 是独立编译的,不会检查跨文件的函数定义是否存在——那是链接器的事。
汇编:把汇编代码转成机器码(目标文件)
汇编器(as)把人类可读的汇编指令(如 mov %rax, %rbx)翻译成二进制机器指令,保存在目标文件(.o 或 .obj)中。目标文件包含代码段(.text)、数据段(.data)、未初始化数据段(.bss),还附带符号表(记录函数名、全局变量名及其地址偏移)和重定位信息(告诉链接器“这里将来要填哪个函数的真实地址”)。
立即学习“C++免费学习笔记(深入)”;
链接:合并多个 .o 文件,解析符号引用
链接器(ld 或 g++ 调用的)把所有目标文件和静态库(.a)按规则合并,并解决“谁调用了谁”的问题。例如 main.o 里有对 add(int,int) 的调用,但没定义;而 math.o 里有它的实现。链接器就把 math.o 中 add 的地址填进 main.o 的调用位置。还会把 printf 等标准函数从动态库(libc.so)中按需加载,生成最终的可执行文件(如 a.out)。
基本上就这些。编译报错(语法/类型)停在第二步,链接报错(undefined reference)停在第四步,运行崩溃(段错误/非法指令)说明前面都过了,问题出在逻辑或内存操作上。










