当前位置: 首页 > news >正文

C及C++编译链接过程详解

一.概述

C/C++程序的构建是一个多阶段的过程,从源代码到最终可执行文件需要经历预处理、编译、汇编和链接四个主要阶段。

二.各个阶段详述

1.预处理阶段 (Preprocessing)

输入:源代码(.c/.cpp) + 头文件(.h)

输出:预处理后的源文件(.i/.ii)

工具:预处理器(cpp)

主要任务:

头文件包含:处理#include指令,将头文件内容插入源文件

宏展开:替换所有#define定义的宏

条件编译:处理#ifdef、#ifndef、#endif等指令

删除注释:移除所有单行(//)和多行(/* */)注释

添加行标记:插入#line指令,用于调试和错误报告

示例命令:

gcc -E main.c -o main.i  # GCC

cl /E main.cpp > main.ii # MSVC

2.编译阶段 (Compilation)

输入:预处理后的源文件(.i/.ii)

输出:汇编代码文件(.s)

工具:编译器(gcc/clang/cl)

主要任务:

词法分析:将源代码分解为标记(tokens)

语法分析:构建抽象语法树(AST)

语义分析:检查类型兼容性、变量声明等

中间代码生成:生成与平台无关的中间表示(IR)

代码优化:进行各种优化(常量折叠、死代码消除等)

目标代码生成:生成特定CPU架构的汇编代码

示例命令:

gcc -S main.i -o main.s  # GCC

cl /FAs main.cpp         # MSVC (生成.asm汇编文件)

3.汇编阶段 (Assembly)

输入:汇编代码文件(.s)

输出:目标文件(.o/.obj)

工具:汇编器(as)

主要任务:

将汇编指令逐条翻译为机器指令

生成目标文件(包含机器码、符号表和重定位信息)

生成节(Section):代码段(.text)、数据段(.data)、BSS段(.bss)等

目标文件结构:

文件头:描述文件属性

节头表:描述各节的位置和属性

.text节:机器指令

.data节:已初始化的全局/静态变量

.bss节:未初始化的全局/静态变量

符号表:函数和变量名及其地址

重定位表:需要链接器修正的地址

示例命令:

as main.s -o main.o      # GCC

cl /c main.cpp           # MSVC (生成.obj文件)

4.链接阶段 (Linking)

输入:目标文件(.o/.obj) + 库文件(.a/.lib)

输出:可执行文件/共享库(.exe/.dll/.so)

工具:链接器(ld/link)

主要任务:

(1) 符号解析 (Symbol Resolution)

解决所有未定义的符号引用

在目标文件和库中查找符号定义

确保每个符号都有唯一明确的定义

(2) 重定位 (Relocation)

合并所有目标文件的相同节

为代码和数据分配最终内存地址

修正代码中的相对地址和绝对地址

(3) 库处理

静态链接:将库代码直接复制到可执行文件中

动态链接:记录库的引用,运行时加载

链接类型对比:

示例命令:

# 静态链接

gcc main.o utils.o -o app -static

# 动态链接

gcc main.o utils.o -o app -lm

# MSVC链接

link main.obj utils.obj /OUT:app.exe

三.关键概念详解

1.符号表 (Symbol Table)

存储所有全局符号(函数、全局变量)

包含符号类型(定义/引用)、大小和位置信息

使用nm工具查看(Unix)或dumpbin /SYMBOLS(Windows)

2.重定位 (Relocation)

修正代码中的地址引用

类型:

PC相对寻址:函数调用、条件跳转

绝对寻址:全局变量访问

重定位信息存储在目标文件的重定位表中

3.名称修饰 (Name Mangling)

C++特有的函数名编码机制

考虑命名空间、类名、参数类型等

目的:支持函数重载

示例:void foo(int) → _Z3fooi

4.静态库 vs 动态库

静态库(.a/.lib):

归档文件(ar命令创建)

本质是目标文件的集合

链接时提取需要的目标文件

动态库(.so/.dll):

包含位置无关代码(PIC)

导出函数表供运行时查找

需要处理符号版本控制和ABI兼容

5.完整构建过程(GCC)

(1)预处理

gcc -E main.c -o main.i

(2)编译

gcc -S main.i -o main.s

(3)汇编

as main.s -o main.o

(4)链接

ld main.o -o app -lc

四.现代构建系统

实际开发中通常使用构建系统管理复杂项目:

http://www.lqws.cn/news/156007.html

相关文章:

  • Devops系列---python基础篇二
  • LSTM-XGBoost多变量时序预测(Matlab完整源码和数据)
  • 【Go】3、Go语言进阶与依赖管理
  • 【VLAs篇】02:Impromptu VLA—用于驱动视觉-语言-动作模型的开放权重和开放数据
  • 【图像处理3D】:世界坐标系
  • React状态管理Context API + useReducer
  • 论文笔记——相干体技术在裂缝预测中的应用研究
  • Spring Boot-面试题(52)
  • Navicat-16.3.9 windows版本 MySQL客户端可视化工具 中文绿色版 无需补丁,无需破解 解压就能用
  • 从Gartner报告看Atlassian在生成式AI领域的创新路径与实践价值
  • 亚马逊:产品被顾客投诉二手产品的申诉模板
  • AI智能推荐实战之RunnableParallel并行链
  • Profinet转CAN网关借助特定配置软件完成子站配置任务
  • 蓝桥杯2118 排列字母
  • 第五届控制与智能机器人国际学术会议(ICCIR 2025)
  • CppCon 2015 学习:C++ in the audio industry
  • AtCoder-abc408_b 解析
  • OD 算法题 B卷【BOSS的收入】
  • 每日Prompt:双重曝光
  • 统信桌面专业版如何使用python开发平台jupyter
  • 鸿蒙jsonToArkTS_工具exe版本来了
  • <论文>(微软)WINA:用于加速大语言模型推理的权重感知神经元激活
  • JupyterNotebook全能指南:从入门到精通
  • Web-图片上传出现的错误
  • 通过Chain Prompts方式将LLM的能力引入测试平台:正交实验测试用例生成
  • 国芯思辰|SCS5501/5502芯片组打破技术壁垒,重构车载视频传输链路,兼容MAX9295A/MAX96717
  • ServBay 1.13.0 更新,新增第三方反向代理/内网穿透
  • 【Android基础回顾】一:Binder机制是什么?有什么用?
  • Kaggle-Predicting Optimal Fertilizers-(多分类+xgboost)
  • npm install 相关命令