cc编译器制作图解
开发C程序有四个步骤:编辑、编译、连接和运行。
任何一个体系结构处理器上都可以使用C语言程序,只要该体系结构处理器有相应的C语言编译器和库,那么C源代码就可以编译并连接到目标二进制文件上运行。
1、预处理:导入源程序并保存(C文件)。
2、编译:将源程序转换为目标文件(Obj文件)。
3、链接:将目标文件生成为可执行文件(EXE文件)。
4、运行:执行,获取运行结果的EXE文件。
(1)cc编译器制作图解扩展阅读:
将C语言代码分为程序的几个阶段:
1、首先,源代码文件测试。以及相关的头文件,比如stdio。H、由预处理器CPP预处理为.I文件。预编译的。文件不包含任何宏定义,因为所有宏都已展开,并且包含的文件已插入。我归档。
2、编译过程是对预处理文件进行词法分析、语法分析、语义分析和优化,生成相应的汇编代码文件。这个过程往往是整个程序的核心部分,也是最复杂的部分之一。
3、汇编程序不直接输出可执行文件,而是输出目标文件。汇编程序可以调用LD来生成可以运行的可执行程序。也就是说,您需要链接大量的文件才能获得“a.out”,即最终的可执行文件。
4、在链接过程中,需要重新调整其他目标文件中定义的函数调用指令,而其他目标文件中定义的变量也存在同样的问题。
2. 鏂版坠钖屽﹀C璇瑷鎺ㄨ崘鍝浜涜蒋浠
鏂版坠瀛︿範C璇瑷涓鑸鐢ㄤ粈涔堣蒋浠
1銆丏ev-C++
Windows 鐜澧冧笅镄勪竴涓杞婚噺绾 C璇瑷/C++ 闆嗘垚寮鍙戠幆澧(IDE)锛岄伒瀹圙PL璁稿彲鍗忚鍒嗗彂婧愪唬镰併傚畠闆嗗悎浜嗗姛鑳藉己澶х殑婧愮爜缂栬緫鍣ㄣ丮ingW64/TDM-GCC 缂栬疟鍣ㄣ丢DB 璋冭瘯鍣ㄥ拰 AStyle 镙煎纺鏁寸悊鍣ㄧ瓑浼楀氲嚜鐢辫蒋浠讹纴镎崭綔鍗佸垎渚挎嵎銆
2銆丆ode::Blocks
杩欐槸涓涓寮婧愩佽法骞冲彴镄凛/C++寮鍙戣蒋浠讹纴涔熸槸杞婚噺绾х殑锛岀浉姣旇缉锷熻兘姣旇缉鍗曚竴镄凞ev-C++𨱒ヨ达纴 Code锛氾细Blocks鏀鎸佸父瑙佺殑镊锷ㄨˉ鍏ㄣ佽娉曟镆ュ拰鎻愮ず镄勫姛鑳介櫎姝や箣澶栵纴杩樿嚜甯︽湁璁稿氭瘆杈冨疄鐢ㄧ殑宸ョ▼妯℃澘锛 鍙浠ュ緢鏂逛究镄勫垱寤箧irectX銆丵t Win32GUI绛夊父瑙佸簲鐢锛 锲犳ゅ紑鍙戞晥鐜囨洿楂桡纴浣跨敤镄勪汉涔熸瘆杈冨
3銆丆Lion
杩欐槸Jet brain鍏鍙歌捐$殑涓涓涓挞棬鐢ㄤ簬C C++寮鍙戠殑闆嗘垚杞浠讹纴鍙浠ヨ法骞冲彴浣跨敤姣旇缉涓扑笟涔熸瘆杈冨己澶э纴鏀鎸佸父瑙佺殑鏅鸿兘琛ュ叏銆佽娉曟镆ャ佷唬镰侀珮浜銆丢it绛夊姛鑳斤纴闄ゆや箣澶栵纴 杩樻敮鎸丆 Make銆佷唬镰侀吨鏋勫拰鍒嗘瀽绛夐珮绾у姛鑳斤纴锲犳ゅ紑鍙戣皟璇曟晥鐜囨洿楂
4銆乂isual Studio
杩椤簲璇ユ槸Windows骞冲彴涓嬩娇鐢ㄦ渶澶氱殑C/C ++寮鍙戣蒋浠朵简锛屽井杞镊涓昏捐$爷鍙戯纴閲嶉噺绾х殑锛 鍙风О瀹囧畽绗涓IDE锛 鐩告瘆杈冨墠闱㈣交閲忕骇镄勫紑鍙戣蒋浠舵潵璇达纴 VisualStudio 椤圭洰绠$悊镟存柟渚匡纴鏀鎸佸崟鍏冩祴璇曪纴浠g爜閲嶆瀯绛夐珮绾у姛鑳斤纴锲犳ゅ湪锲㈤槦鍗忎綔锛屽ぇ鍨嬭蒋浠跺紑鍙戜腑缁忓父浼氱敤鍒帮纴鍒濇湡浣跨敤浼氭湁涓瀹氶毦搴︼纴浣嗘槸镡熸倝鎺屾彙钖庯纴镄勭‘鏄鏄涓涓狢/C++镄勫紑鍙戝埄鍣
5銆乂isual Studio Code
杩欐槸涓涓寮婧愩佽法骞冲彴镄勪唬镰佺紪杈戝櫒锛屼篃鏄寰杞镄勪骇鍝侊纴 涓ユ牸镒忎箟涓婃潵璇达纴 VS Co 涔愯█鍙浠ュ綋锅氢竴涓闱炲父涓嶉敊镄凛璇瑷缂栫▼杞浠舵潵浣跨敤锛岃交渚跨伒娲汇佽繍琛岄熷害蹇銆佸崰鐢ㄥ唴瀛桦皯甯歌佺殑镊锷ㄨˉ鍏锛岃娉曟彁绀篏it 绛夊姛鑳介兘鑳藉緢濂芥敮鎸侊纴鎻掍欢镓╁𪾢涓板瘜锛岀敓镐佺幆澧冭坛濂斤纴锲犳わ纴瀵逛簬鏂版坠𨱒ヨ达纴涔熸槸涓涓闱炲父涓嶉敊镄勯夋嫨锛 钖屾剧殑SublimeText绛夌紪杈戝櫒涔熸湁闱炲父涓嶉敊
3. cc鍦ㄧ紪绋嬩腑鏄浠涔堟剰镐
CC鍦ㄧ紪绋嬩腑鏄疌璇瑷镄勭缉鍐欙纴瀹冩槸涓绉嶉氱敤缂栫▼璇瑷锛岀粡甯歌鐢ㄤ簬绯荤粺缂栫▼銆佸祵鍏ュ纺绯荤粺銆佺绣缁灭紪绋嫔拰娓告垙寮鍙戠瓑浠诲姟銆侰璇瑷镄勮捐$亩娲併侀珮鏁堬纴琚骞挎硾璁ゅ彲涓哄彲绉绘嶆у己銆佽祫婧愭秷钥椾绠镄勭紪绋嬭瑷銆傚畠鏄疞inux镎崭綔绯荤粺镄勫熀纭锛屼篃琚骞挎硾搴旂敤浜庡叾浠栨搷浣灭郴缁熺幆澧冧腑銆
闄や简C璇瑷镄勭缉鍐欙纴CC鍦ㄧ紪绋嬩腑甯稿父琚鐢ㄤ綔缂栬疟鍣ㄧ殑浠gО銆傜紪璇戝櫒鏄涓绉嶅皢婧愪唬镰佽浆鎹㈡垚鍙镓ц屼唬镰佺殑宸ュ叿銆傚畠灏嗘簮浠g爜缈昏疟鎴愭満鍣ㄨ瑷锛屼娇寰楄$畻链鸿兘澶熺悊瑙e拰镓ц岀▼搴忋侰C缂栬疟鍣ㄦ渶鍒濇槸鐢盇T&T镄刄nix锲㈤槦寮鍙戠殑锛屼篃琚绉颁负Unix C缂栬疟鍣ㄣ傜幇鍦锛孋C缂栬疟鍣ㄥ凡缁忔垚涓烘爣鍑嗙殑C缂栬疟鍣锛屽箍娉涘簲鐢ㄤ簬涓嶅悓镄勭紪绋嬬幆澧冧腑銆
闄や简C璇瑷鍜岀紪璇戝櫒锛孋C鍦ㄧ紪绋嬩腑杩樻湁鍏朵粬镄勫惈涔夈备緥濡傦纴瀹冨彲浠ヤ唬琛–reative Commons锛岃繖鏄涓涓镊村姏浜庢帹锷ㄧ煡璇嗗叡浜镄勭粍缁囥侰reative Commons涓哄垱浣滆呮彁渚涗简涓浜涢氱敤镄勭増𨱒冭稿彲𨱒℃撅纴浣垮缑鍏朵粬浜哄彲浠ュ湪涓瀹氭浔浠朵笅浣跨敤鍜屽垎浜浠栦滑镄勪綔鍝併傚湪涓浜涘紑婧愯蒋浠跺拰缃戠粶璧勬簮涓锛屾垜浠鍙浠ョ湅鍒癈C鎺堟潈镄勬爣璇嗭纴杩栾〃鏄庣潃鐩稿簲镄勫垱浣灭墿镄勭増𨱒冨凡缁忚鍏变韩鍑烘潵浜嗐
4. 如何制作arm-linux-gcc编译工具
一、下载源文件
源代码文件及其版本:
binutils-2.19.tar.bz2, gcc-core-4.4.4.tar.bz2 gcc-g++-4.4.4.tar.bz2 Glibc-2.7.tar.bz2 Glibc-ports-2.7.tar.bz2 Gmp-4.2.tar.bz2 mpfr-2.4.0.tar.bz2mpc-1.0.1.tar.gz Linux-2.6.25.tar.bz2 (由于我在编译出错的过程中,根据出错的信息修改了相关的C代码,故而没有下载相应的补丁)
一般一个完整的交叉编译器涉及到多个软件,主要包括bilinguals、cc、glibc等。其中,binutils主要生成一些辅助工具;gcc是用来生成交叉编译器,主要生成arm-linux-gcc交叉编译工具,而glibc主要提供用户程序所需要的一些基本函数库。
二、建立工作目录
编译所用主机型号 fc14.i686,虚拟机选的是VM7.0,Linux发行版选的是Fedora9,
第一次编译时用的是root用户(第二次用一般用户yyz), 所有的工作目录都在/home/yyz/cross下面建立完成,首先在/home/yyz目录下建立cross目录,然后进入工作目录,查看当前目录。命令如下:
创建工具链文件夹:
[root@localhost cross]# mkdir embedded-toolchains
下面在此文件夹下建立如下几个目录:
setup-dir:存放下载的压缩包;
src-dir:存放binutils、gcc、glibc解压之后的源文件;
Kernel:存放内核文件,对内核的配置和编译工作也在此完成;
build-dir :编译src-dir下面的源文件,这是GNU推荐的源文件目录与编译目录分离的做法;
tool-chain:交叉编译工具链的安装位;
program:存放编写程序;
doc:说明文档和脚本文件;
下面建立目录,并拷贝源文件。
[root@localhost cross] #cd embedded- toolchains
[root@localhost embedded- toolchains] #mkdir setup-dir src-dir kernel build-dir tool-chain program doc
[root@localhost embedded- toolchains] #ls
build-dir doc kernel program setup-dir src-dir tool-chain
[root@localhost embedded- toolchains] #cd setup-dir
拷贝源文件:
这里我们采用直接拷贝源文件的方法,首先应该修改setup-dir的权限
[root@localhost embedded- toolchains] #chmod 777 setup-dir
然后直接拷贝/home/yyz目录下的源文件到setup-dir目录中,如下图:
建立编译目录:
[root@localhost setup-dir] #cd ../build-dir
[root@localhost build -dir] #mkdir build-binutils build-gcc build-glibc
三、输出环境变量
输出如下的环境变量方便我们编译。
为简化操作过程。下面就建立shell命令脚本environment-variables:
[root@localhost build -dir] #cd ../doc
[root@localhost doc] #mkdir scripts
[root@localhost doc] #cd scripts
用编辑器vi编辑环境变量脚本envionment-variables:[root@localhost scripts]
#vi envionment-variables
export PRJROOT=/home/yyz/cross/embedded-toolchains
export TARGET=arm-linux
export PREFIX=$PRJROOT/tool-chain
export TARGET_PREFIX=$PREFIX/$TARGET
export PATH=$PREFIX/bin:$PATH
截图如下:
执行如下语句使环境变量生效:
[root@localhost scripts]# source ./environment-variables
四、建立二进制工具(binutils)
下面将分步介绍安装binutils-2.19.1的过程。
[root@localhost script] # cd $PRJROOT/src-dir
[root@localhost src-dir] # tar jxvf ../setup-dir/binutils-2.19.1.tar.bz2
[root@localhost src-dir] # cd $PRJROOT/build-dir/build-binutils
创建Makefile:
[root@localhost build-binutils] #../../src-dir/binutils-2.19.1/configure --target=$TARGET --prefix=$PREFIX
在build-binutils目录下面生成Makefile文件,然后执行make,make install,此过程比较缓慢,大约需要一个15分钟左右。完成后可以在$PREFIX/bin下面看到我们的新的binutil。
输入如下命令
[root@localhost build-binutils]#ls $PREFIX/bin
5. 什么是GCC编译器
Linux系统下的Gcc(GNU C Compiler)是GNU推出的功能强大、性能优越的多平台编译器,是GNU的代表作品之一。gcc是可以在多种硬体平台上编译出可执行程序的超级编译器,其执行效率与一般的编译器相比平均效率要高20%~30%。
Gcc编译器能将C、C++语言源程序、汇程式化序和目标程序编译、连接成可执行文件,如果没有给出可执行文件的名字,gcc将生成一个名为a.out的文件。在Linux系统中,可执行文件没有统一的后缀,系统从文件的属性来区分可执行文件和不可执行文件。而gcc则通过后缀来区别输入文件的类别,下面我们来介绍gcc所遵循的部分约定规则。
.c为后缀的文件,C语言源代码文件;
.a为后缀的文件,是由目标文件构成的档案库文件;
.C,.cc或.cxx 为后缀的文件,是C++源代码文件;
.h为后缀的文件,是程序所包含的头文件;
.i 为后缀的文件,是已经预处理过的C源代码文件;
.ii为后缀的文件,是已经预处理过的C++源代码文件;
.m为后缀的文件,是Objective-C源代码文件;
.o为后缀的文件,是编译后的目标文件;
.s为后缀的文件,是汇编语言源代码文件;
.S为后缀的文件,是经过预编译的汇编语言源代码文件。
Gcc的执行过程
虽然我们称Gcc是C语言的编译器,但使用gcc由C语言源代码文件生成可执行文件的过程不仅仅是编译的过程,而是要经历四个相互关联的步骤∶预处理(也称预编译,Preprocessing)、编译(Compilation)、汇编(Assembly)和连接(Linking)。
命令gcc首先调用cpp进行预处理,在预处理过程中,对源代码文件中的文件包含(include)、预编译语句(如宏定义define等)进行分析。接着调用cc1进行编译,这个阶段根据输入文件生成以.o为后缀的目标文件。汇编过程是针对汇编语言的步骤,调用as进行工作,一般来讲,.S为后缀的汇编语言源代码文件和汇编、.s为后缀的汇编语言文件经过预编译和汇编之后都生成以.o为后缀的目标文件。当所有的目标文件都生成之后,gcc就调用ld来完成最后的关键性工作,这个阶段就是连接。在连接阶段,所有的目标文件被安排在可执行程序中的恰当的位置,同时,该程序所调用到的库函数也从各自所在的档案库中连到合适的地方。
Gcc的基本用法和选项
在使用Gcc编译器的时候,我们必须给出一系列必要的调用参数和文件名称。Gcc编译器的调用参数大约有100多个,其中多数参数我们可能根本就用不到,这里只介绍其中最基本、最常用的参数。
Gcc最基本的用法是∶gcc [options] [filenames]
其中options就是编译器所需要的参数,filenames给出相关的文件名称。
-c,只编译,不连接成为可执行文件,编译器只是由输入的.c等源代码文件生成.o为后缀的目标文件,通常用于编译不包含主程序的子程序文件。
-o output_filename,确定输出文件的名称为output_filename,同时这个名称不能和源文件同名。如果不给出这个选项,gcc就给出预设的可执行文件a.out。
-g,产生符号调试工具(GNU的gdb)所必要的符号资讯,要想对源代码进行调试,我们就必须加入这个选项。
-O,对程序进行优化编译、连接,采用这个选项,整个源代码会在编译、连接过程中进行优化处理,这样产生的可执行文件的执行效率可以提高,但是,编译、连接的速度就相应地要慢一些。
-O2,比-O更好的优化编译、连接,当然整个编译、连接过程会更慢。
-Idirname,将dirname所指出的目录加入到程序头文件目录列表中,是在预编译过程中使用的参数。C程序中的头文件包含两种情况∶
A)#include
B)#include “myinc.h”
其中,A类使用尖括号(< >),B类使用双引号(“ ”)。对于A类,预处理程序cpp在系统预设包含文件目录(如/usr/include)中搜寻相应的文件,而对于B类,cpp在当前目录中搜寻头文件,这个选项的作用是告诉cpp,如果在当前目录中没有找到需要的文件,就到指定的dirname目录中去寻找。在程序设计中,如果我们需要的这种包含文件分别分布在不同的目录中,就需要逐个使用-I选项给出搜索路径。
-Ldirname,将dirname所指出的目录加入到程序函数档案库文件的目录列表中,是在连接过程中使用的参数。在预设状态下,连接程序ld在系统的预设路径中(如/usr/lib)寻找所需要的档案库文件,这个选项告诉连接程序,首先到-L指定的目录中去寻找,然后到系统预设路径中寻找,如果函数库存放在多个目录下,就需要依次使用这个选项,给出相应的存放目录。
-lname,在连接时,装载名字为“libname.a”的函数库,该函数库位于系统预设的目录或者由-L选项确定的目录下。例如,-lm表示连接名为“libm.a”的数学函数库。
上面我们简要介绍了gcc编译器最常用的功能和主要参数选项,更为详尽的资料可以参看Linux系统的联机帮助。
假定我们有一个程序名为test.c的C语言源代码文件,要生成一个可执行文件,最简单的办法就是∶
gcc test.c
这时,预编译、编译连接一次完成,生成一个系统预设的名为a.out的可执行文件,对于稍为复杂的情况,比如有多个源代码文件、需要连接档案库或者有其他比较特别的要求,就要给定适当的调用选项参数。再看一个简单的例子。
整个源代码程序由两个文件testmain.c 和testsub.c组成,程序中使用了系统提供的数学库,同时希望给出的可执行文件为test,这时的编译命令可以是∶
gcc testmain.c testsub.c □lm □o test
其中,-lm表示连接系统的数学库libm.a。
Gcc的错误类型及对策
Gcc编译器如果发现源程序中有错误,就无法继续进行,也无法生成最终的可执行文件。为了便于修改,gcc给出错误资讯,我们必须对这些错误资讯逐个进行分析、处理,并修改相应的语言,才能保证源代码的正确编译连接。gcc给出的错误资讯一般可以分为四大类,下面我们分别讨论其产生的原因和对策。
第一类∶C语法错误
错误资讯∶文件source.c中第n行有语法错误(syntex errror)。这种类型的错误,一般都是C语言的语法错误,应该仔细检查源代码文件中第n行及该行之前的程序,有时也需要对该文件所包含的头文件进行检查。有些情况下,一个很简单的语法错误,gcc会给出一大堆错误,我们最主要的是要保持清醒的头脑,不要被其吓倒,必要的时候再参考一下C语言的基本教材。
第二类∶头文件错误
错误资讯∶找不到头文件head.h(Can not find include file head.h)。这类错误是源代码文件中的包含头文件有问题,可能的原因有头文件名错误、指定的头文件所在目录名错误等,也可能是错误地使用了双引号和尖括号。
第三类∶档案库错误
错误资讯∶连接程序找不到所需的函数库,例如∶
ld: -lm: No such file or directory
这类错误是与目标文件相连接的函数库有错误,可能的原因是函数库名错误、指定的函数库所在目录名称错误等,检查的方法是使用find命令在可能的目录中寻找相应的函数库名,确定档案库及目录的名称并修改程序中及编译选项中的名称。
第四类∶未定义符号
错误资讯∶有未定义的符号(Undefined symbol)。这类错误是在连接过程中出现的,可能有两种原因∶一是使用者自己定义的函数或者全局变量所在源代码文件,没有被编译、连接,或者干脆还没有定义,这需要使用者根据实际情况修改源程序,给出全局变量或者函数的定义体;二是未定义的符号是一个标准的库函数,在源程序中使用了该库函数,而连接过程中还没有给定相应的函数库的名称,或者是该档案库的目录名称有问题,这时需要使用档案库维护命令ar检查我们需要的库函数到底位于哪一个函数库中,确定之后,修改gcc连接选项中的-l和-L项。
排除编译、连接过程中的错误,应该说这只是程序设计中最简单、最基本的一个步骤,可以说只是开了个头。这个过程中的错误,只是我们在使用C语言描述一个算法中所产生的错误,是比较容易排除的。我们写一个程序,到编译、连接通过为止,应该说刚刚开始,程序在运行过程中所出现的问题,是算法设计有问题,说得更玄点是对问题的认识和理解不够,还需要更加深入地测试、调试和修改。一个程序,稍为复杂的程序,往往要经过多次的编译、连接和测试、修改。下面我们学习的程序维护、调试工具和版本维护就是在程序调试、测试过程中使用的,用来解决调测阶段所出现的问题。窗体顶端
窗体底端
6. 如何在linux环境下编辑一个c语言源程序并将其编译为可执行文件
打开安装好的Linux系统(ubuntu18.04)
1.安装编辑器
使用语句:sudo apt-get install vim安装vim,可以在安装前查看是否安装vim编辑器。(特别提示:在安装vim前最好将软件更新,使用指令sudo apt-get update.)之所以要加sudo的原因是这些安装和更新软件的时候需要使用管理员权限才可以进行。
2.安装gcc编译器
和安装vim类似使用sudo apt-get install gcc,安装完成后可以使用cc-v来查看编译器版本信息等。cc -v界面如下:
7. Linux下如何编译,运行C程序需要安装编译器吗
首先一定要安装 gcc (或者 cc )编译器。然后在 Linux 系统下,首先使用 vi 全屏幕编辑程序编辑一个后缀名为 .c 的文件,然后使用 gcc 编译器对你的 C 语言源程序进行编译、连接。最后才能够运行生成后的运行文件(如果你的源程序没有任何编译错误的话)。当然了,在 gcc 的编译过程中,会有很多选择项。这个就是靠编程经验了。
举例如下:
gcc test.c -o myrunfile ( -o 选项指定输出的运行文件名为:myrunfile,如果不指定 -o 选项的话,缺省的运行文件名为:a.out)