• GCC 使用介绍


    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修饰(画蛇添足)而引起的。
  • 相关阅读:
    requests库简单介绍与使用
    python爬虫之无界面谷歌浏览器介绍
    PhantomJS介绍与使用
    使用find_elements_by_class_name定位元素有时候会出现打印出来的列表里面数据为空的现象,解决方案
    【C#】多态
    【JavaScrpt】JS之数组去重
    【SQL】sql语句在insert一条记录后返回该记录的ID
    【SQL】SQL整表复制
    【C#】获取URL上的参数
    【C#】 break continue return 的区别
  • 原文地址:https://www.cnblogs.com/legendbaby/p/5057069.html
Copyright © 2020-2023  润新知