gcc编译器的整个工作过程

gcc hello.c   ./a.out     或者 gcc hello.c -o hello   ./hello

./表示执行当前目录下的可执行程序或脚本程序。

首先gcc需要调用预处理程序cpp,由它负责展开在源文件中定义的宏,并向其中插入“#include”语句所包含的内容;接着gcc会调用编译程序ccl和汇编程序as将处理后的源代码编译成目标代码;最后,gcc会调用链接程序ld,把生成的目标代码链接成一个可执行程序。其实gcc本身只是做了编译这一项工作,其余阶段的工作都是gcc调用其余服务程序来完成的。

预处理阶段。gcc把预处理命令扫描处理完毕,输入C语言的源文件(.c),这个阶段主要处理源文件中的#ifdef、#include、#define等预处理命令,该阶段会生成一个中间文件.i。可以使用-E参数让gcc在预处理阶段结束后停止编译过程,从而生成经过预处理的C源代码文件:gcc -E hello.c –o hello.i  vim hello.i  //查看该文件实际的变化      该阶段详细情况举例说明说明如下:

//head.h                              
#ifndef __HEAD_H__
#define __HEAD_H__#define NUM1 10
#define NUM2 20
#endif//sum.c
#include <stdio.h>  //直接在标准库中查找
#include "head.h"   //先在工作目录中查找,找不到再去标准库中查找
#define DEBUG     //去掉这一行,gcc编译时采用-D参数即可,生成最终文件时,不用-D参数。
int main(void)
{int a = NUM1;int aa;int b = NUM2;int sum = a + b;// 小盆友: 这是一个加法运算
#ifdef DEBUGprintf("The sum value is: %d + %d = %d\n", a, b, sum);
#endifreturn 0;
}

如上段代码中,有两个文件,一个头文件head.h和一个c语言源代码文件sum.c。执行gcc -E sum.c -o sum.i后,打开sum.c可以看到如下内容:

//这上面还有很多内容,全是stdio.h头文件的内容
extern int ftrylockfile (FILE *__stream) __attribute__ ((__nothrow__ , __leaf__)) ;extern void funlockfile (FILE *__stream) __attribute__ ((__nothrow__ , __leaf__));
# 943 "/usr/include/stdio.h" 3 4# 2 "sum.c" 2
# 1 "head.h" 1
# 3 "sum.c" 2int main(void)
{int a = 10;int aa;int b = 20;int sum = a + b;return 0;
}

可以看到在预处理过程中,预处理程序主要做了以下几件事情:1.处理#include,将所有头文件的内容都全部包含进来;2.处理掉所有的条件预编译命令,#ifdef   #ifndef  #endif等;3.删除所有的#define,并且展开所有的宏定义,即字符替换;4.删除所有注释;5.添加行号和文件标识,这样在调试和编译出错时才知道是哪个文件哪一行的问题;5.保留#pragma编译器指令,因为编译器在编译过程中需要使用它们。

#pragma para(其中para为参数)编译器指令的作用是设定编译器的状态或者是指示编译器完成一些特定的动作。#pragma 指令对每个编译器给出了一个方法,在保持与C 和C ++语言完全兼容的情况下,给出主机或操作系统专有的特征。依据定义,编译指示是机器或操作系统专有的,且对于每个编译器都是不同的。

编译阶段。gcc把预处理后的结果编译成汇编语言代码,输入的是.i,编译后生成汇编语言文件.s:gcc -S hello.i –o hello.s  vim hello.s    hello.s为汇编语言代码(没学过汇编的表示也看不懂!!)

汇编阶段。编译器把编译出来的汇编语言汇编成具体CPU上的目标代码(机器代码)。输入汇编代码文件.s,输出目标代码文件.o或.obj:gcc –c hello.s –o hello.o  vim hello.o  .o文件也是一个二进制代码文件,只是还不能执行,需要进行链接。

链接阶段。把多个目标代码模块链接成一个大的目标代码模块。输入目标代码文件.0(与其它的目标代码文件、库文件、引导代码),汇集成一个可执行的二进制代码文件:  gcc hello.o –o hello  vim hello       执行: ./hello    

后续(操作系统范畴):机器代码hello在操作系统机器上解释操作系统的;然后在机器语言机器上被翻译称为一个个微程序;最后,微程序的每一条微指令在微指令系统上执行。

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

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

相关文章

宏定义对调试代码的作用

以如下代码为例&#xff1a; //head.h #ifndef __HEAD_H__ #define __HEAD_H__#define NUM1 10 #define NUM2 20 #endif//sum.c #include <stdio.h> //直接在标准库中查找 #include "head.h" //先在工作目录中查找&#xff…

【第15章】多重继承

1. 虚基类介绍 多继承时很容易产生命名冲突&#xff0c;即使我们很小心地将所有类中的成员变量和成员函数都命名为不同的名字&#xff0c;命名冲突依然有可能发生&#xff0c;比如非常经典的菱形继承层次。如下图所示&#xff1a; 类A派生出类B和类C&#xff0c;类D继承自类B和…

gcc编译器与g++编译器的区别

gcc与g编译器的程序文件分别为&#xff1a;/usr/bin/g和/usr/bin/gcc。 gcc 和 GCC 是两个不同的东西&#xff0c;GCC:GNU Compiler Collection(GUN 编译器集合)&#xff0c;它可以编译C、C、JAV、Fortran、Pascal、Object-C、Ada等语言。gcc是GCC中的GUN C Compiler&#xff0…

1. 排序算法

一、概述 假定在待排序的记录序列中&#xff0c;存在多个具有相同的关键字的记录&#xff0c;若经过排序&#xff0c;这些记录的相对次序保持不变&#xff0c;即在原序列中&#xff0c;r[i]r[j]&#xff0c;且r[i]在r[j]之前&#xff0c;而在排序后的序列中&#xff0c;r[i]仍…

1036. 跟奥巴马一起编程(15)

美国总统奥巴马不仅呼吁所有人都学习编程&#xff0c;甚至以身作则编写代码&#xff0c;成为美国历史上首位编写计算机代码的总统。2014年底&#xff0c;为庆祝“计算机科学教育周”正式启动&#xff0c;奥巴马编写了很简单的计算机代码&#xff1a;在屏幕上画一个正方形。现在…

库文件与头文件

首先说明库文件与头文件在gcc中的具体使用方法&#xff0c;然后说明两者的区别与联系。 库文件即库函数&#xff0c;如printf和scanf函数。以libgtdf.so库文件为例&#xff08;库文件在命名时都以lib开头&#xff0c;因此使用-l选项去链接指定的库文件时可以省略lib三个字母&am…

gcc的常用参数

-c 编译成目标文件.o&#xff08;只编译不链接&#xff09; gcc -c hello.s -o hello.o -o 指出输出文件名&#xff0c;输出文件名跟在-o后面。如果不使用这一选项&#xff0c;则缺省的输出文件名为a.out。gcc hello.c -o hello.exe&#xff08;在Linux中该项后缀名无要求&a…

1027. 打印沙漏(20)

本题要求你写个程序把给定的符号打印成沙漏的形状。例如给定17个“*”&#xff0c;要求按下列格式打印 ************ *****所谓“沙漏形状”&#xff0c;是指每行输出奇数个符号&#xff1b;各行符号中心对齐&#xff1b;相邻两行符号数差2&#xff1b;符号数先从大到小顺序递减…

【C++ Priemr | 15】构造函数与拷贝控制

继承的构造函数 1. 简介&#xff1a; 子类为完成基类初始化&#xff0c;在C11之前&#xff0c;需要在初始化列表调用基类的构造函数&#xff0c;从而完成构造函数的传递。如果基类拥有多个构造函数&#xff0c;那么子类也需要实现多个与基类构造函数对应的构造函数。 class …

C命令行参数

C命令行参数的作用是在执行程序时&#xff0c;可以将命令行的参数传值给C程序内部&#xff0c;这样就可以从外部控制程序&#xff0c;而不是在代码内对这些值进行硬编码。命令行参数是使用main函数来处理的&#xff0c;argc是指参数的个数&#xff0c;为int类型&#xff1b;arg…

剖析数组名、函数名(不是指针常量,更不是指针)

对于一个数组&#xff0c;如 int a[4]; 如果只是给出数组名a&#xff0c;编译器不知道该取该数组的第几个元素&#xff0c;因此编译器不会自动取值&#xff0c;而是返回该数组的首地址&#xff08;第一个元素的地址&#xff09;。其实&#xff0c;数组名a就是数组本身&#xf…

【C++ Priemr | 15】面向对象程序设计

类型准换与继承 为了支持c的多态性&#xff0c;才用了动态绑定和静态绑定。 需要理解四个名词&#xff1a; 对象的静态类型&#xff1a;对象在声明时采用的类型&#xff0c;是在编译期确定的。对象的动态类型&#xff1a;目前所指对象的类型&#xff0c;是在运行期决定的。对…

linux里source、. 、sh、bash、./有什么区别

转载&#xff1a;https://www.cnblogs.com/pcat/p/5467188.html 1.source a.sh source可以简写为“.”&#xff0c;即. a.sh 注意中间有空格&#xff0c;在当前shell内去读取、执行a.sh&#xff0c;而a.sh不需要有"执行权限"。 2.sh a.sh 和 bash a.sh 都是打开…

【C++ Priemr | 15】虚函数表剖析(三)

一、虚拟菱形继承 #include <iostream> using namespace std;class B { public:int _b; };class C1 :virtual public B { public:int _c1; };class C2 :virtual public B { public:int _c2; };class D :public C1, public C2 { public:int _d; };int main() {cout <&…

gcc的警告提示信息

gcc包含完整的出错检查和警告提示功能。采用-pedantic选项&#xff0c;对于不符合ANSI/ISO标准的源代码会产生相应的警告信息。如&#xff1a;gcc -pedantic hello.c -o hello (main函数返回类型为int&#xff0c;且函数体内要有return 语句&#xff0c;一般为 return 0;) -pe…

1037. 在霍格沃茨找零钱(20)

如果你是哈利波特迷&#xff0c;你会知道魔法世界有它自己的货币系统 —— 就如海格告诉哈利的&#xff1a;“十七个银西可(Sickle)兑一个加隆(Galleon)&#xff0c;二十九个纳特(Knut)兑一个西可&#xff0c;很容易。”现在&#xff0c;给定哈利应付的价钱P和他实付的钱A&…

【Leetcode | 6】136. 只出现一次的数字

给定一个非空整数数组&#xff0c;除了某个元素只出现一次以外&#xff0c;其余每个元素均出现两次。找出那个只出现了一次的元素。 说明&#xff1a; 你的算法应该具有线性时间复杂度。 你可以不使用额外空间来实现吗&#xff1f; 示例 1: 输入: [2,2,1] 输出: 1 示例 2: 输入…

gcc的优化功能

代码优化的目的是改善程序的执行性能。gcc提供的代码优化功能非常强大&#xff0c;它通过参数-On来控制优化代码的生成&#xff0c;其中n为优化级别的整数&#xff0c;比较典型的范围是从0变化到2或3&#xff08;与版本有关&#xff09;。 编译时通过使用选项-O可以告诉gcc同时…

gcc编译多个源代码文件的过程(引出makefile)

由foo1.c foo2.c foo3.c 3个源文件组成的源程序生成最终的可执行程序foo的命令&#xff1a; gcc foo1.c foo2.c foo3.c -o foo 如果处理的源文件不止一个&#xff0c;则gcc会依次对每个文件进行预处理、编译、汇编&#xff0c;最后将所有的目标代码和库文件进行&#xff0c;链…

观擦者模式

/********************************************************************created: 2006/07/20filename: Observer.hauthor: 李创http://www.cppblog.com/converse/purpose: Observer模式的演示代码 *********************************************************************/…