【C++】预编译、编译、汇编、链接

Windows中以2:2划分内核、用户空间,Linux中以1:3划分内核、用户空间。 

在text段中,只有普通局部变量是指令 

int gdata1 = 10; //.data  已初始化且初始化不为零的数据
int gdata2 = 0;  //.bss   未初始化或初始化为零的数据
int gdata3;      //.bss
static int gdata4 = 20;  //.data
static int gdata5 = 0;   //.bss
static int gdata6;       //.bss
int main()
{
	int data1 = 30; //.text  //只有普通局部变量是指令
	int data2 = 0;  //.text
	int data3;      //.text
	static int data4 = 40; //.data
	static int data5 = 0;  //.bss
	static int data6;      //.bss
	
	return 0;
}

预编译:i

1.#define  宏文本替换

2.#include  递归展开头文件

3.#if #endif #elif  删除预编译指令

4.删除注释

5.添加行号和文本标识

6.保留 #pragma  编译器处理

编译:s  【编译阶段以一个.cpp或者.c文件为单元编译】

1.词法分析

2.语法分析

3.语义分析

4.代码优化  生成汇编代码

[inter x86]  //汇编代码
int a = 10;
mov dword ptr[a],0Ah //ptr[a] 对a地址解引用后得到a的内存 //0Ah 是10

汇编:o  可重定位(重入)的二进制文件

  把指令代码翻译成二进制

  Linux中ELF文件,bss段:段以符号起始,节省空间

  bss段少了一个数据,和虚拟地址空间上的段不同意义。 (bss段中少的数据,位于COM块中)

  强弱符号:强符号是已初始化的全局变量,弱符号是未初始化的全局变量【C++中无强弱符号之分】

  强弱符号规则:

1.两强:重定义错误  //数据段不可以出现重名

2.一强一弱:选强符号作为所有地址

3.两弱:选字节数大的   <编译器处理>

在汇编完成前,不清楚是否存在强符号无法判断时,则将变量放入COM块中。

链接: 生成.exe文件  也是ELF文件,有头和.data、.text段等

          //UND  未定义区  //找不到

1.段合并:相同段合并<一个段映射一个页面> 

   合并符号表:同名查找,未找到则用本身查找的弱符号,找到则删除弱符号改用强符号

2.符号解析(处理UND):未找到对应的符号进行报错     合并UND

3.分配地址和空间

4.符号的重定位  //test段  <虚假地址改真实,纠正虚假偏移>

链接代码:[ld -e main -o run main.o sum.o]

【Linux】中关于预编译、编译、汇编、链接的代码

1.[预编译] gcc  -E  Aff.c  -o  main.i

    [ls]  Aff.c   main.i

2.[编译]gcc  -S  main.i  -o  main.s

    [ls]Aff.c   mian.i   main.s

3.[汇编]gcc  -C  mian.s  -o  main.o

    [ls]Aff.c   mian.i   main.s   main.o   a.out

4.[链接]/a.out    //此时是.exe文件(可执行文件)

猜你喜欢

转载自blog.csdn.net/weixin_40740059/article/details/84075653