Gcc 的使用

  编译器在编译过程中,先将程式码编译成 object 档,然後再和程式库联结,成为可执行档。故一个编译器须提供的参数主要有几类:
  1.指定编译器编出的object 档或是可执行档档名。 
  2.在编译过程做最佳化,可提升程式的执行速度。 
  3.设定搜寻程式库的标头档 (header file) 及程式库档的目录及指定程式库档档名。
 以下便以这四个大类分别介绍。 
  注意:下面在不同类别中所介绍的参数,几乎都可以混合著使用。
 
参数: -o out_put_filename
 
  .设定编译出的 object 档档名或是可执行档档名: 
  说明: 指定编译出的档名为 out_put_filename。 
  范例: 本例将程式码 'test.c' 编译成可执行档,并设定档名为 'test'。 
  gcc test.c -o test
 
参数: -O
 
. 在编译过程做最佳化 
  说明: 在编译过程做最佳化,以提升增快程式执行速度。
-O0
-O1
-O2
-O3
编译器的优化选项的4个级别,-O0表示没有优化,-O1为缺省值,-O3优化级别最高  
  范例: 本例将程式码 'test.c' 编译成可执行档 'test',并在编译过程做最佳化'。 
  gcc -O test.c -o test
 
参数: -include和-I
 
-include用来包含头文件,但一般情况下包含头文件都在源码里用#include xxxxxx实现,-include参数很少用。-I参数是用来指定头文件目录,/usr/include目录一般是不用指定的,gcc知道去那里找,但是如果头文件不在/usr/include里我们就要用-I参数指定了,比如头文件放在/myinclude目录里,那编译命令行就要加上-I/myinclude参数了,如果不加你会得到一个"xxxx.h: No such file or directory"的错误。-I参数可以用相对路径,比如头文件在当前目录,可以用 -I.来指定。上面我们提到的--cflags参数就是用来生成-I参数的
  
参数: -Ldir_name
 
设定搜寻程式库目录
说明: 将目录 'dir_name' 设定为搜寻程式库目录之一。
放在/lib和/usr/lib和/usr/local/lib里的库直接用-l参数就能链接了,但如果库文件没放在这三个目录里, 而是放在其他目录里,这时我们只用-l参数的话,链接还是会出错,出错信息大概是:“/usr/bin/ld: cannot find -lxxx”,也就是链接程序ld在那3个目录里找不到libxxx.so,这时另外一个参数-L就派上用场了,比如 
常用的X11的库,它放在/usr/X11R6/lib目录下,我们编译时就要用-L/usr/X11R6/lib -lX11参数,-L参数跟 着的是库文件所在的目录名。再比如我们把libtest.so放在/aaa/bbb/ccc目录下,那链接参数就是-L/aaa/bbb /ccc -ltest
 
参数: -lname
  
设定程式库档案。 
说明: 联结程式库 libname.a 。 
-l参数就是用来指定程序要链接的库,-l参数紧接着就是库名,那么库名跟真正的库文件名有什么关系呢? 就拿数学库来说,他的库名是m,他的库文件名是libm.so,很容易看出,把库文件名的头lib和尾.so去掉就 是库名了。 
好了现在我们知道怎么得到库名了,比如我们自已要用到一个第三方提供的库名字叫libtest.so,那么我们 只要把libtest.so拷贝到/usr/lib里,编译时加上-ltest参数,我们就能用上libtest.so库了(当然要用 libtest.so库里的函数,我们还需要与libtest.so配套的头文件)。 

另外,大部分libxxxx.so只是一个链接,以RH9为例,比如libm.so它链接到/lib/libm.so.x,/lib/libm.so.6 又链接到/lib/libm-2.3.2.so,如果没有这样的链接,还是会出错,因为ld只会找libxxxx.so,所以如果你要 用到xxxx库,而只有libxxxx.so.x或者libxxxx-x.x.x.so,做一个链接就可以了ln -s libxxxx-x.x.x.so libxxxx.so
 
参数: -Wall
  说明: 输出较多的警告讯息,以便找出程式的错误。
  范例: 编译 test.c 时输出较多的警告讯息。
  gcc -Wall test.c
 
参数: -g
  说明: 在编译出可执行档时,附加执行时除错资讯,以供 gdb 读取 (若要使用 ABSoft 的除错程式,则须将参数改为 -gdwarf )。 
  范例: 将 'test.c' 编译成可执行档 'test',并附加除错资讯。 
  gcc -g test.c -o test
 
参数: -c
      仅编译成 object 档。
  说明: 仅编译成 object 档而不进行程式库联结。 
  范例: 将 test.c 编译成 object 档 test.o。 
  gcc -c test.c -o test.o
 
联结数个 object 成可执行档。
  范例: 将 'test1.o'、'test2.o' 和程式库联结後成为可执行档 test。 
  gcc test1.o test2.o -o test
 
参数: -E
      观察宏展开情形:
  说明:展开程式中的宏以便了解巨集是否依照预期方式展开。 
  范例:将 test1.c 中的宏展开後储存到 test1.c.ext 。 
  gcc -E test1.c > test1.c.ext
 
参数: -S
      产生组合语言程式码:
  范例:编译 test.c 产生对应的组合语言程式码档 test.s。 
  gcc -S test.c -o test.s
 
参数:-ansi
  关闭gnu c中与ansi c不兼容的特性,激活ansi c的专有特性(包括禁止一 些asm inline typeof关键字,以及UNIX,vax等预处理宏,
 
参数:-D 定义宏(D-define)
    -D定义宏有两种情况,一种是-DMACRO 相当于程序中使用#define MACRO 另外可以-DMACRO=A 相当于程序中使用#define MACRO A 这只是一个编绎参数,在连接时没有意义 
如: $gcc -c hello.c -o hello.o -DDEBUG 
    上面为hello.c定义了一个DEBUG宏,某些情况下使用-D 代替直接在文件中使用#define,也是为了避免修改源代码双。例如一个程序希望在开发调试的时候能打印出调试信息,而正式发布的时候就不用打印了,而且发布前不用修改源代码双。可以这样 
#ifdefine DEBUG 
printf("debug message/n"); 
#endif 
    对于这段代码,平时调试的时候就加上-DDEBUG 发布时不用-D选项 
与之对应的是-UMACRO参数,相当于#undef MACRO,取消宏定义
 
参数:-C
  在预处理的时候,不删除注释信息,一般和-E使用,有时候分析程序,用这个很
  方便的
 
参数:M
  生成文件关联的信息。包含目标文件所依赖的所有源代码
  你可以用gcc -M hello.c来测试一下,很简单。
 
参数:-MM -MD -MMD 
    和上面的那个一样,但是它将忽略由#i nclude造成的依赖关系。
-MD
  和-M相同,但是输出将导入到.d的文件里面
-MMD
  和-MM相同,但是输出将导入到.d的文件里
 
-static
  此选项将禁止使用动态库,所以,编译出来的东西,一般都很大,也不需要什么
动态连接库,就可以运行.
 
-share
  此选项将尽量使用动态库,所以生成文件比较小,但是需要系统由动态库.
 
-traditional
  试图让编译器支持传统的C语言特性
 
-gstabs –gstabs+ -ggdb
-gstabs
此选项以stabs格式声称调试信息,但是不包括gdb调试信息.
 -gstabs+
  此选项以stabs格式声称调试信息,并且包含仅供gdb使用的额外调试信息  
-ggdb
  此选项将尽可能的生成gdb的可以使用的调试信息.
 
参数:-Wa,option     -wl.option
-Wa,option
此选项传递option给汇编程序;如果option中间有逗号,就将option分成多个选项,然后传递给会汇编程序 
-Wl.option
  此选项传递option给连接程序;如果option中间有逗号,就将option分成多个选项,然后传递给会连接程序.
 
交叉编译器的使用方法
使用方法跟本地的gcc差不多,但有一点特殊的是:必须用-L和-I参数指定编译器用sparc系统的库和头文件,不能用本地(X86) 的库(头文件有时可以用本地的)。 
例子: sparc-xxxx-linux-gnu-gcc test.c -L/path/to/sparcLib -I/path/to/sparcInclude
 
几个相关的环境变量
PKG_CONFIG_PATH:用来指定pkg-config用到的pc文件的路径,默认是/usr/lib/pkgconfig,pc文件是文本文件,扩展名是
.pc,里面定义开发包的安装路径,Libs参数和Cflags参数等等。
CC:用来指定c编译器。
CXX:用来指定cxx编译器。
LIBS:跟上面的--libs作用差不多。
CFLAGS:跟上面的--cflags作用差不多。
CC,CXX,LIBS,CFLAGS手动编译时一般用不上,在做configure时有时用到,一般情况下不用管。
环境变量设定方法:export ENV_NAME=xxxxxxxxxxxxxxxxx
 
Gcc编译常见问题
 
为什么会出现undefined reference to 'xxxxx'错误?
 
首先这是链接错误,不是编译错误,也就是说如果只有这个错误,说明你的程序源码本身没有问题,是你 
用编译器编译时参数用得不对,你没有指定链接程序要用到得库,比如你的程序里用到了一些数学函数, 
那么你就要在编译参数里指定程序要链接数学库,方法是在编译命令行里加入-lm。
 
 错误"dereferencing pointer to incomplete type"- -
typedef struct {
   int data;
   char type;
}STR;

int main(){
   struct STR * pstr;
   pstr->data=0;
}
上面已经使用typdef 来定义 STR了,但是在主函数里面定义pstr的时候仍然使用了struct修饰(画蛇添足)而引起的。
 posted on 2015-12-18 14:48  legend.baby  阅读(692)  评论(1编辑  收藏  举报