当前位置:  技术问答>linux和unix

怎么在linux中编译与运行c++的原代码*.cpp文件

    来源: 互联网  发布时间:2015-06-02

    本文导语:  怎么在linux中编译与运行c++的原代码*.cpp文件(在终端中不是在kdevelop中) 最好举个例子(有详细参数介绍),谢谢!!!!!!! | gcc 简介  gcc简介  Linux系统下的gcc(GNU C Compiler)是GNU推出...

怎么在linux中编译与运行c++的原代码*.cpp文件(在终端中不是在kdevelop中)
最好举个例子(有详细参数介绍),谢谢!!!!!!!

|
gcc 简介 
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,这个过程可以用图12-1框图描述。 


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语言描述一个算法中所产生的错误,是比较容易排除的。我们写一个程序,到编译、连接通过为止,应该说刚刚开始,程序在运行过程中所出现的问题,是算法设计有问题,说得更玄点是对问题的认识和理解不够,还需要更加深入地测试、调试和修改。一个程序,稍为复杂的程序,往往要经过多次的编译、连接和测试、修改。下面我们学习的程序维护、调试工具和版本维护就是在程序调试、测试过程中使用的,用来解决调测阶段所出现的问题。 

|
[介绍] 
gcc and g++分别是gnu的c & c++编译器 

gcc/g++在执行编译工作的时候,总共需要4步 

1.预处理,生成.i的文件 
2.将预处理后的文件不转换成汇编语言,生成文件.s 
3.有汇编变为目标代码(机器代码)生成.o的文件 
4.连接目标代码,生成可执行程序 


[参数详解] 
-x language filename  
   设定文件所使用的语言,使后缀名无效,对以后的多个有效.也就是根 
   据约定C语言的后缀名称是.c的,而C++的后缀名是.C或者.cpp,如果 
   你很个性,决定你的C代码文件的后缀名是.pig 哈哈,那你就要用这 
   个参数,这个参数对他后面的文件名都起作用,除非到了下一个参数 
   的使用。 
   可以使用的参数吗有下面的这些 
     `c', `objective-c', `c-header', `c++', `cpp-output',  
     `assembler', and `assembler-with-cpp'. 
   看到英文,应该可以理解的。 
   例子用法: 
   gcc -x c hello.pig 
   
-x none filename 
  关掉上一个选项,也就是让gcc根据文件名后缀,自动识别文件类型 
  例子用法: 
  gcc -x c hello.pig -x none hello2.c 
   
-c  
  只激活预处理,编译,和汇编,也就是他只把程序做成obj文件 
  例子用法: 
  gcc -c hello.c 
  他将生成.o的obj文件  

-S 
  只激活预处理和编译,就是指把文件编译成为汇编代码。 
  例子用法 
  gcc -S hello.c 
  他将生成.s的汇编代码,你可以用文本编辑器察看 

-E 
  只激活预处理,这个不生成文件,你需要把它重定向到一个输出文件里 
  面. 
  例子用法: 
  gcc -E hello.c > pianoapan.txt 
  gcc -E hello.c | more 
  慢慢看吧,一个hello word 也要与处理成800行的代码 

-o 
  制定目标名称,缺省的时候,gcc 编译出来的文件是a.out,很难听,如果 
  你和我有同感,改掉它,哈哈 
  例子用法 
  gcc -o hello.exe hello.c (哦,windows用习惯了) 
  gcc -o hello.asm -S hello.c 

-pipe 
  使用管道代替编译中临时文件,在使用非gnu汇编工具的时候,可能有些问 
  题 
  gcc -pipe -o hello.exe hello.c 

-ansi 
  关闭gnu c中与ansi c不兼容的特性,激活ansi c的专有特性(包括禁止一 
  些asm inline typeof关键字,以及UNIX,vax等预处理宏, 

-fno-asm 
  此选项实现ansi选项的功能的一部分,它禁止将asm,inline和typeof用作 
  关键字。 
     
-fno-strict-prototype 
  只对g++起作用,使用这个选项,g++将对不带参数的函数,都认为是没有显式 
  的对参数的个数和类型说明,而不是没有参数. 
  而gcc无论是否使用这个参数,都将对没有带参数的函数,认为城没有显式说 
  明的类型 
   
-fthis-is-varialble 
  就是向传统c++看齐,可以使用this当一般变量使用. 
   
-fcond-mismatch 
  允许条件表达式的第二和第三参数类型不匹配,表达式的值将为void类型 
   
-funsigned-char 
-fno-signed-char 
-fsigned-char 
-fno-unsigned-char 
  这四个参数是对char类型进行设置,决定将char类型设置成unsigned char(前 
  两个参数)或者 signed char(后两个参数) 
   
-include file 
  包含某个代码,简单来说,就是便以某个文件,需要另一个文件的时候,就可以 
  用它设定,功能就相当于在代码中使用#include 
  例子用法: 
  gcc hello.c -include /root/pianopan.h 
   
-imacros file 
  将file文件的宏,扩展到gcc/g++的输入文件,宏定义本身并不出现在输入文件 
  中 
   
-Dmacro 
  相当于C语言中的#define macro 
   
-Dmacro=defn 
  相当于C语言中的#define macro=defn 
   
-Umacro 
  相当于C语言中的#undef macro 

-undef 
  取消对任何非标准宏的定义 
   
-Idir 
  在你是用#include"file"的时候,gcc/g++会先在当前目录查找你所制定的头 
  文件,如果没有找到,他回到缺省的头文件目录找,如果使用-I制定了目录,他 
  回先在你所制定的目录查找,然后再按常规的顺序去找. 
  对于#include,gcc/g++会到-I制定的目录查找,查找不到,然后将到系 
  统的缺省的头文件目录查找 
   
-I- 
  就是取消前一个参数的功能,所以一般在-Idir之后使用 
   
-idirafter dir 
  在-I的目录里面查找失败,讲到这个目录里面查找. 
   
-iprefix prefix 
-iwithprefix dir 
  一般一起使用,当-I的目录查找失败,会到prefix+dir下查找 
   
-nostdinc 
  使编译器不再系统缺省的头文件目录里面找头文件,一般和-I联合使用,明确 
  限定头文件的位置 
   
-nostdin C++ 
  规定不在g++指定的标准路经中搜索,但仍在其他路径中搜索,.此选项在创建 
  libg++库使用 
   
-C 
  在预处理的时候,不删除注释信息,一般和-E使用,有时候分析程序,用这个很 
  方便的 
   
-M 
  生成文件关联的信息。包含目标文件所依赖的所有源代码 
  你可以用gcc -M hello.c来测试一下,很简单。 
   
-MM 
  和上面的那个一样,但是它将忽略由#include造成的依赖关系。 
   
-MD 
  和-M相同,但是输出将导入到.d的文件里面 
   
-MMD 
  和-MM相同,但是输出将导入到.d的文件里面 
   
-Wa,option 
  此选项传递option给汇编程序;如果option中间有逗号,就将option分成多个选 
  项,然后传递给会汇编程序 
   
-Wl.option 
  此选项传递option给连接程序;如果option中间有逗号,就将option分成多个选 
  项,然后传递给会连接程序. 
   

-llibrary  
  制定编译的时候使用的库 
  例子用法 
  gcc -lcurses hello.c 
  使用ncurses库编译程序 
   
-Ldir 
  制定编译的时候,搜索库的路径。比如你自己的库,可以用它制定目录,不然 
  编译器将只在标准库的目录找。这个dir就是目录的名称。 
   
-O0 
-O1 
-O2 
-O3 
  编译器的优化选项的4个级别,-O0表示没有优化,-O1为缺省值,-O3优化级别最 
  高   
   
-g 
  只是编译器,在编译的时候,产生条是信息。 
   
-gstabs 
  此选项以stabs格式声称调试信息,但是不包括gdb调试信息. 
   
-gstabs+ 
  此选项以stabs格式声称调试信息,并且包含仅供gdb使用的额外调试信息. 
   
-ggdb 
  此选项将尽可能的生成gdb的可以使用的调试信息. 

    
 
 

您可能感兴趣的文章:

  • Linux c++虚函数(virtual function)简单用法示例代码
  • 为什么linux下的C++程序这么少见? 请问那里有linux下的C++程序?什么类型的程序都可以.
  • Linux c++库boost unordered_set数据插入及查找代码举例
  • linux的帮助文件在什么地方啊,我想看看关于c++的帮助文件。另外从哪里能搞到c++的类库说明阿
  • unix/Linux下c++ boost thread库读写锁介绍
  • 装好了ubuntu怎样学习c++,linux下C++学习的环境软件是什么?
  • Linux c++库boost unordered_map数据插入及查找代码举例
  • stlport c++库如何移植到arm-linux平台替代gnu c++标准库
  • Linux c++ boost库正则表达式用法
  • 求教linux C++学习之路!
  • Windows和Linux下C++类成员方法作为线程函数方法介绍
  • C++在linux下是闲得蛋疼?
  • Linux 下c++开发error while loading shared libraries问题解决
  • Linux下怎样编译C++程序?
  • Linux和windows下用mysql c++ library操作Mysql数据库
  • 关于Linux下C++开发的问题
  • linux菜鸟请教:C++的学习
  • linux下的C++编译器怎样使用?
  • Linux C++ 网络库 Muduo
  • 请问,在Linux和Unix下,有没有像Borland C++ 3.1 for DOS或Visual C++ 6.0这样的开发工具?
  • C++在linux下意味者什么
  • linux/centos源码安装nginx编译配置选项参数介绍
  • 请问如何在linux下运行windows游戏? iis7站长之家
  • linux 64位编译出错。linux 32位可以编译过。
  • 关于在linux交叉编译powerpc-e300c3-linux-gcc的编译问题
  • Linux下gcc编译时,如何以静态链接的形式编译?
  • linux下有没有能编译出16bit代码的C语言编译器?
  • 现要在一台Pentium100上安装LINUX操作系统,能不能不选用别的品牌的LINUX,而直接用内核编译,然后再加上编译器和SHELL等等其它东西?如
  • linux下可以编译c但是如何编译c++
  • arm-linux-gcc交叉编译出错,但gcc编译没错,什么原因呢???
  • 谁编译过PLX9054 Linux的驱动程序,我给100分教我编译和安装??
  •  
    本站(WWW.)旨在分享和传播互联网科技相关的资讯和技术,将尽最大努力为读者提供更好的信息聚合和浏览方式。
    本站(WWW.)站内文章除注明原创外,均为转载、整理或搜集自网络。欢迎任何形式的转载,转载请注明出处。












  • 相关文章推荐
  • Linux下指定运行时加载动态库路径及shell下执行程序默认路径
  • 弱弱地问,能否让WINDOWS下自动运行的多媒体光盘在LINUX下也自动运行播放(LINUX下可一看到光盘上文件)
  • 我的linux程序 如何 在pc机器上运行。现在我用arm-linux-g++ 编译后的程序在我的嵌入设备上运行。
  • 程序在windows下用visual运行正常,如何才能在linux下运行??
  • 怎样让linux启动后不运行桌面而是直接运行某个应用程序呢?
  • Linux下的程序是在内存中运行的吗?为什么在程序运行的时候可以删除程序文件?
  • linux编写一个脚本判断程序是否在运行,如果没运行就重启这个软件。
  • 请问如何让Linux 0.11运行?它运行后是怎样的?谢谢!!
  • 新手提问:grep在linux中运行跟在php运行返回结果不一样
  • Linux开关机时想让脚本运行却运行不了?
  • 谁有linux?帮我用在linux运行两个文件好吗?
  • linux环境下,java程序手工运行不乱码,crontab自动运行就乱码
  • vmware怎么优化?在vmware上装linux运行起来是不是特别慢?我在256内存的p4上安装,运行起来很慢!
  • LINUX 下使用QT开发的程序如何才能让它在没有安装QT的Linux系统下运行?
  • 请问在高版本linux编译链接的程序如何在低版本Linux上运行
  • linux运行级别
  • 我在linux下安装了linux版的realone,但却没法找到如何运行。
  • 在Linux Shell scripts下变量如何取linux的命令运行值
  • 请问如何在linux下运行windows游戏?
  • 在linux中什么叫运行级别
  • 初学者提问:可以在Linux上运行exe程序吗?
  • linux c/c++ IP字符串转换成可比较大小的数字
  • 在win分区上安装linux和独立分区安装linux有什么区别?可以同时安装吗?(两个linux系统)
  • linux哪个版本好?linux操作系统版本详细介绍及选择方案推荐
  • 在虚拟机上安装的linux上,能像真的linux系统一样开发linux程序么?
  • secureCRT下Linux终端汉字乱码解决方法
  • 我重装window后,把linux的引导区覆盖了,进不了linux怎么办?急啊,望热心的人帮助 (现在有linux的盘)
  • Linux c字符串中不可打印字符转换成16进制
  • 安装vmware软件,不用再安装linux系统,就可以模拟linux系统了,然后可以在其上学习一下LINUX下的基本操作 了?
  • Linux常用命令介绍:更改所属用户群组或档案属性
  • 红旗Linux主机可以通过127.0.0.1访问,但如何是连网的Win2000机器通过Linux的IP去访问Linux


  • 站内导航:


    特别声明:169IT网站部分信息来自互联网,如果侵犯您的权利,请及时告知,本站将立即删除!

    ©2012-2021,,E-mail:www_#163.com(请将#改为@)

    浙ICP备11055608号-3