linux4-gcc编译链接、执行、链接过程、gdb调试

1.gcc如何安装,g++同理

在这里插入图片描述
在这里插入图片描述

2.gcc编译链接

底层原理:
//mian.c文件
(1)预编译(语法错误等也能通过)
gcc -E main.c -o main.i
(2)编译
gcc -S main.i -o main.s
(3)汇编
gcc -C main.s -o main.o
(4)链接
gcc main.o -o main

在这里插入图片描述
在这里插入图片描述

3.执行:

如何启动一个进程:
路径+可执行文件名
路径+可执行文件名 &
在这里插入图片描述

4.编译链接过程

编译链接原理:
main.c 通过预编译生成main.i 文件;
main.i 通过编译生成 main.s 文件;
main.s 通过汇编生成 main.o 文件;
链接就是将所有的 .o 文件 .a文件(静态库文件) .lib文件(库文件) .obj文件 链接起来,生成 .out 文件(ELF格式的可执行文件)
Windows是生成 .exe 文件
在这里插入图片描述
最早运行程序时,时以二进制指令0101…等基础代码编译运行,但很繁琐。为了方便使用,对特定的一些固定功能的代码给以统称表示,称为汇编助记符,如mov 、add。但不能跨平台。再后来,出现高级语言,如C、C++等。但机器不能理解运行,需要重新将程序转成机器语言。

gcc main.o 默认输出a.out
在这里插入图片描述
(重要,能回答基本问题)

(1)预编译阶段:

a)删除所有的"#define",并且展开所有的宏定义;
b)处理所有的条件预编译指令,“#if”,“#ifdef”,“#endif"等;
c)处理”#include"预编译指令,将被包含的文件插入到该预编译指令的位置;
d)删除所有的注释;
e)添加行号和文件名标识,以便于编译器产生调试用的符号信息及编译时产生编译错
误和警告时显示行号;
f)保留所有的#pragma编译器指令,因为编译器需要使用它们;

(2)编译阶段

词法分析,语法分析,语意分析,代码优化,汇总符号;

(3)汇编阶段

将汇编指令翻译成二进制格式,生成各个section,生成符号表

(4)链接阶段

a)合并各个section,调整section的起始位置和段大小,合并符号表,进行符号解析,给符
号分配虚拟地址;
b)符号重定位,即在使用符号的地方全部替换成符号的虚拟地址;

编译链接原理:书-程序员的自我修养

源代码,main.c
在这里插入图片描述
预编译:main.i
文件内容多,是吧#include <stdio.h>的内容贴近来了。语法出错也能通过
在这里插入图片描述
编译:main.s
在这里插入图片描述
汇编:main.o
ELF格式的可执行文件
在这里插入图片描述

5.执行

(1)关于执行:路径+可执行文件名

为什么基础指令例如 ls,可以直接运行,不需要加路径?
因为ls在标准路径里有, – /usr/bin
在这里插入图片描述
所以可以将可执行文件放到 /usr/bin 下,也可直接运行:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
可以把可执行文件放到 /usr/bin 就可以省略路径

(2)关于一步执行(重点)

四步合为一步:
gcc -o main main.c

在这里插入图片描述

(3)关于两步执行

三步合为一步:
gcc -c main.c -o main.o 或者 gcc -c main.c
(可以先编译,看看代码编译能不能通过)
gcc -o main main.o
在这里插入图片描述

(4)补充:gcc可以编译c++文件

gcc -o mainc++(名字) mainc++.cpp(名字) -lstdc++(指明用c++的标准库)

6.gdb调试

(1)debug版本

在编译阶段会加入某些调试信息;
调试信息是在编译的过程中加入到中间文件.o文件的;

(2)release版本(发行版本,体积小,没有调试信息)

gcc 默认生成的是release版本

(3)安装gdb

(4)加入调式信息:

gcc -o test test.c -g (在编译阶段加的调试信息,非链接阶段)
进入调试:
gdb test
在这里插入图片描述
错误代码:
在这里插入图片描述
修改后:
(第12行代码错误,键盘读入为:‘end\n’,所以比较前三项)
在这里插入图片描述

在这里插入图片描述

(5)常用的调试命令:

(熟悉)
加断点: b+行号
启动程序:r(先加断点再启动)
显示代码: l
显示断点信息: info break/info b
删除断点信息: delete 断点编号
单步执行: n
打印: p
自动打印: display
退出: q

进入函数:s
跳出函数: finish
b+函数名: 加断点到函数入口处
取消一个display :undisplay display 的编号
c:继续(continue)

|
在这里插入图片描述
在这里插入图片描述
加断点: b+行号
启动程序: r

在这里插入图片描述
单步执行: n
在这里插入图片描述
显示断点信息: info break/info b在这里插入图片描述

gdb命令(相对:全):
l:显示main函数所在的文件的源代码
list 文件名:num 显示文件名文件num行上下的源代码
b 行号:给指定行添加断点
b 函数名:给指定函数的第一有效行添加一个断点
info break:显示断点信息;
delete 断点号:删除指定断点
r(run):运行程序
n(next):单步执行
c(continue):继续执行,直接执行到下一个断点处
s:进入将要被调用的函数中执行
finish:跳出函数;
q:退出调试
bt:显示函数调用栈
disable 断点号:将断点设定为无效的,不加断点号,将所有断点设置为无效
enable 断点号:将断点设定为有效的,不加断点号,将所有断点设置为有效;
p val:打印变量val的值
p &val:打印变量val的地址
p a+b:打印表达式的值
p arr(数组名):打印数组所有元素的值
*parr@len:用指向数组的指针打印数组所有元素的值
display:自动显示,参数和p命令一样;
info display:显示自动显示信息
undisplay+编号:删除指定的自动显示
ptype val:显示变量类型

1.练习gdb调试;
2.练习多文件的编译:

//main.c 
#include <stdio.h> 
int add(int x,int y); 
int max(int x,int y); 

int main() 
{
    int a=5,b=10;
    printf("a+b=%d\n",add(a,b));
    printf("max=%d\n",max(a,b));
    
    return 0; 
 }
 
 //add.c 
 int add(int x,int y) 
 {
     return x+y; 
 }
 
//max.c 
int max(int x,int y) 
{ 
    return x>y?x:y; 
}
//多文件的编译 
gcc -o main main.c add.c max.c 


gcc -c main.c 
gcc -c add.c 
gcc -c max.c 
//生成了.o文件 
gcc -o main main.o add.o max.o

在这里插入图片描述
在这里插入图片描述

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值