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 1003 4 int main()5 {6     printf("hello %d\n ", M);7 8     //条件编译9 #ifdef DEBUG 10     printf("debug\n");11 #else12     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秒一打印
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/505243.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ABAP - SALV教程04 添加状态栏

CL_SALV_TABLE中提供了 GET_FUNCTIONS方法.GET_FUNCTIONS方法返回的是一个CL_SALV_FUNCTIONS_LIST类型的实例对象. 类CL_SALV_FUNCTIONS_LIST两个方法(SET_ALL、SET_DEFAULT)可以添加标准状态栏 实现步骤: 定义SET_STATUS私有方法. PRIVATE SECTION.METHODS:set_status CHA…

Revit-二开之创建墙-(6)

Revit API窗间墙 protected override Result OnExecute(ExternalCommandData commandData, ref string message, ElementSet elements) {try{// 获取当前活动的文档

qsort使用

qsort 是用来排序的数据的库函数,底层使用的是快速排序的方式 排序方式有:选择,冒泡,插入,快速, 希尔...... 对于qsort这个库函数: void qsort(void* base,size_t num,size_t size,int (*compar)(const void*,const void*) 其中 void* base 是指针,指向的是待排序的数组的第…

Unity 向量计算、欧拉角与四元数转换、输出文本、告警、错误、修改时间、定时器、路径、

using System.Collections; using System.Collections.Generic; using UnityEngine;public class c2 : MonoBehaviour {// 定时器float t1 0;void Start(){// 向量Vector3 v1 new Vector3(0, 0, 2);Vector3 v2 new Vector3(0, 0, 3);// 计算两个向量的夹角Debug.Log(Vector3…

基于java+springboot女士电商平台系统源码+文档设计

基于javaspringboot女士电商平台系统源码文档设计 博主介绍&#xff1a;多年java开发经验&#xff0c;专注Java开发、定制、远程、文档编写指导等,csdn特邀作者、专注于Java技术领域 作者主页 央顺技术团队 Java毕设项目精品实战案例《1000套》 欢迎点赞 收藏 ⭐留言 文末获取源…

光伏业务管理软件都有哪些?

随着光伏行业的快速发展&#xff0c;光伏业务管理软件在光伏企业的日常运营和管理中扮演着越来越重要的角色。这些软件不仅提高了光伏企业的运营效率&#xff0c;还帮助企业更好地管理项目、优化资源配置、降低成本并提升市场竞争力。本文将介绍一些常见的光伏业务管理软件&…

41、网络编程/TCP.UDP通信模型练习20240301

一、编写基于TCP的客户端实现以下功能&#xff1a; 通过键盘按键控制机械臂&#xff1a;w(红色臂角度增大)s&#xff08;红色臂角度减小&#xff09;d&#xff08;蓝色臂角度增大&#xff09;a&#xff08;蓝色臂角度减小&#xff09;按键控制机械臂 1.基于TCP服务器的机械臂…

ELK学习

ELK 一、ELK介绍 &#x1f604; “ELK”是三个开源项目的首字母缩写&#xff0c;这三个项目分别是&#xff1a;Elasticsearch、Logstash 和 Kibana。Elasticsearch 是一个搜索和分析引擎。Logstash 是服务器端数据处理管道&#xff0c;能够同时从多个来源采集数据&#xff0…

软考重点题解析-基础知识

1.加密技术&#xff1a;分为对称加密技术&#xff1a;文件的加密和解密使用相同的密钥 和 非对称加密技术&#xff1a;加密和解密不同的密钥&#xff0c;分别是公开密钥和私有密钥。 例题&#xff1a;若A,B两人分别在认证机构&#xff08;CA&#xff09;M,N处获得证书&…

每日leetcode--最大数

正题之前 三玖yyds&#xff01;&#xff01;&#xff01; 题目 给定一组非负整数 nums&#xff0c;重新排列每个数的顺序&#xff08;每个数不可拆分&#xff09;使之组成一个最大的整数。 注意&#xff1a;输出结果可能非常大&#xff0c;所以你需要返回一个字符串而不是整…

STM32 | J-link安装过程

J-Link是一种由SEGGER公司开发的调试器和仿真器,用于嵌入式系统开发。它可以帮助开发人员在开发过程中进行调试和仿真,提供了快速、稳定的连接,并支持多种不同类型的微处理器和微控制器。 要获取J-Link软件,请访问SEGGER公司的官方网站(www.segger.com)并进入他们的下载…

解决:code ERESOLVE:ERESOLVE could not resolve 的报错问题

报错实例 报错原因 是我执行npm i xxx-xx的时候会出现这个错误 查了资料表示是node.js的问题 或者的依赖本身的问题 解决 1.在后面加上 --legacy-peer-deps 示例&#xff1a;npm i sass-loader7.3.1 --legacy-peer-deps 2&#xff0c;检查node版本&#xff0c;更改node版本 …