07 编译器

目录

  1. 编译过程
  2. 编译器查看
  3. 详解
  4. 函数库
  5. 自动化构建工具
  6. 进度条程序

1. 编译过程

预处理: a. 去注释 b.宏替换 c.头文件展开 d.条件编译
编译: 汇编
汇编: 可重定向二进制目标文件
链接: 链接多个.o, .obj合并形成一个可执行exe
gcc编译c程序, g++编译c++程序

2. 编译器查看

输入gcc -v , g++ -v ,查看编译器的版本,如果不存在g++,可以用命令安装

sudo yum install -y gcc-c++

在这里插入图片描述

3. 详解

如果要编译成可执行文件,首先编写一个.c文件,然后输入:

gcc [选项] 要编译的文件 [选项] [目标文件]

例如: gcc code.c -o myexe , -o后面带要生成的程序名
在这里插入图片描述

3.1 预处理

预处理功能主要包括宏定义、文件包含展开,条件编译,去注释等
预处理指令是以#号开头的代码行
实例: gcc -E hello.c -o hello.i
E的作用是让预处理后停止编译,选项-o是生成目标文件,不然会打印在屏幕上,.i为预处理后的文件

准备要一份这样的源代码:

  1 #include <stdio.h>
  2 #define M 100
  3 
  4 int main()
  5 {
  6     printf("hello %d\n ", M);
  7 
  8     //条件编译
  9 #ifdef DEBUG 
 10     printf("debug\n");
 11 #else
 12     printf("release\n");
 13 #endif                                                                                        
 14     return 0;
 15 }

执行预处理结果:

在这里插入图片描述
打开源代码和生成的.i文件对比
在这里插入图片描述

上面的840行才到main函数,前面的一大段部分是include头文件展开的结果,里面都是头文件函数的声明。define定义的宏M直接用100替换了输出的地方。条件编译这四个注释的字也去了,因为gcc默认编译为release模式,所以条件编译只保留条件成立的结果

3.2 编译

这个阶段首先检查代码的规范性,语法错误以及实际要做的工作,确保无误后翻译成汇编语言

实例:

gcc -S [.i文件] -o [文件名.s]

源文件翻译完成停止编译,将上面的.i 文件编译成.s文件查看
在这里插入图片描述在这里插入图片描述

3.3 汇编

生成机器识别的二进制代码,实例:

gcc -c hello.s -o hello.o

将上面的汇编文件转换为二进制目标文件查看

在这里插入图片描述在这里插入图片描述
由于是二进制文件,编辑器不能识别,所以显示为乱码

3.4 链接

生成可执行文件或库文件
实例:

gcc hello.o -o hello

4. 函数库

我们的程序中并没有printf函数的实现,而且预处理后也只有函数的声明,那是在哪里实现的printf函数

输入 file [程序名]
在这里插入图片描述
这里说动态链接分享的库,答案是这些函数的实现都在名为libc.so.6的库文件中,gcc会在路径/usr/bin目录下寻找,就能链接到库函数

/usr/include目录下有函数头文件,提供c语言函数的声明
在这里插入图片描述

ldd [程序]可以看到依赖的库,提供函数的实现
在这里插入图片描述

库分为动态库和静态库
静态库是编译链接时,把库文件的代码全部加入到可执行文件中,因此生成的文件比较大,运行时也不需要库文件了
动态库是一个包含可由多个程序同时使用的代码和数据的库,动态链接提供了一种方法地址,程序执行时去找对应的函数实现
win:.dll 是动态库 .lib是静态库
linux: .so是动态库 .a是静态库

静态库需要拷贝函数实现所以文件更大,动态库生成的程序更小
gcc/g++默认动态链接的,输入命令使用静态链接

gcc [文件] -o [程序名] -static

在这里插入图片描述
静态链接的文件体积大很多

如果没有静态库,可以安装

sudo yum install -y glibc-static
sudo yum install -y libstdc+±static

gcc选项

-E 只激活预处理,不生成文件,需要重定向到输出文件
-S 编译到汇编语言不进行汇编和链接
-c 编译到目标代码
-o 文件输出到文件
-static 此选项仅对生成的文件采用静态链接
-g 生成调试信息,GNU调试器可利用该信息
-shared 尽量使用动态库,生成文件较小,需要有动态库
-O0
-O1
-O2
-O3 编译器优化的四个级别,0表示没有优化,1为缺省值,3最高
-w 不生成任何警告信息
-Wall 生成所有警告信息

可以记忆为编译选项Esc ,生成iso文件

5. 自动化构建

make和makefile

一个工程中的源文件不计其数,放在若干个目录中,makefile定义了一系列规则,哪些
文件需要先编译,哪些文件后编译,什么时候重新编译
带来的好处就是自动化编译,一旦写好,以后只需要调用,极大提高了开发效率。
make是一个命令工具,解释makefile指令的工具,一般来说大多数ide都有这个命令,如:Delphi的make,vc的nmake,linux下GNU的make
make是命令,makefile是文件,两个搭配使用

自动化构建的makefile需要明确两个东西,依赖关系和依赖方法,依赖关系自顶向下匹配,所以把首先把依赖关系根部的写最前面

准备三个文件,一个头文件和实现,一个main函数
头文件,add.h
在这里插入图片描述

实现 add.c
在这里插入图片描述

main文件 main.c
在这里插入图片描述

先用gcc编译一下程序有没有错误

在这里插入图片描述
创建一个makefile文件,用来自动化构建
在这里插入图片描述
mytest是要生成的程序名,依赖于add.o和main.o两个文件,这两个文件各自又依赖于源文件,gcc生成这两个源文件,最后加clean的依赖关系和依赖方法,删除所有临时文件和程序

输入make就会自动生成程序
在这里插入图片描述
输入make clean就会清楚项目
在这里插入图片描述

当多次make会显示程序已经是最新的了
在这里插入图片描述

这是因为程序如果已经是最新的,就没必要继续生成。makefile如何得知程序是最新的?。前面说过文件有acm时间,a是访问时间,因为是频繁操作所以达到一定量会刷新,c是内容修改时间,m是文件属性修改时间,一般内容修改,文件大小也会变化。一般情况下,生成的程序的修改时间应该是最晚的,make时会对比依赖的两个源文件是否有内容变化,如果这两个文件的修改时间晚于程序的时间,就说明内容有变化,这时可以make生成

我们修改一下main文件的参数重新make
在这里插入图片描述
修改后就可以构建了

.PHONY的依赖关系总是被执行的,这个如何理解
当我们构建时,如果程序是最新的将无法生成,但我们执行clean却不受时间限制,每次都可以执行,这就是总是可以执行

在写项目时,makefile先确保没问题再写项目

6. 进度条程序

\r和\n两个是不同的功能,在最初的计算机中,\n用来换行,而光标还在上一行的位置,\r回车用来重新回到本行开头,所以到下一行开头就是\r\n

#include <stdio.h>    
    
int main()    
{    
    printf("hello \r");    
    return  0;                                                                                     
}    

输入上面程序执行
发现什么都没有打印,这是因为\r让光标回到了最开头所以没有文本。在计算机里,输出并不是立即输出,有一个输出缓冲区的概念,有自己的刷新规则。碰到\n会把\n之前的内容刷新到目标设备。也可以用一些强制刷新缓冲区的函数,如:fflush(stdout)函数

制作进度条,每次打印增加的部分和百分比,都在同一行不断刷新,不换行。这个可以用\r打印

#include <stdio.h>    
#include <unistd.h>    
       
int main()    
{      
    int cnt = 1;    
    char str[100]= {0};    
    while(cnt <=50)    
    {    
    str[cnt - 1] = '#';    
    printf("[进度][%-50s](%d%%) \r", str,cnt*2);    
    fflush(stdout);    
    usleep(30000);    
    cnt++;    
    }     
       
    printf("\n");                                                                               
    return 0;                                                
}  

由于屏幕不够,每行打印50个扩大2倍作为百分百,usleep可以查询,是一个延时的函数,单位是微秒,上面是0.3秒一打印
在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值