目录

即看即用

一、GCC 简介

GCC编译流程

GCC编译选项

二、GCC生成动态库和静态库

三、生成动态库和静态库实例

四、静态库和动态库的使用和配置

静态库的使用

共享库的使用

不到你指定链接的so错误

编译链接静态库、动态库

链接动态库


即看即用

多个源文件/目标生成动态库
a.
gcc -fPIC -shared xxx1.c xxx2.c xxx3.c -o libxxx.so 
b.
gcc -fPIC -shared xxx1.o xxx2.o xxx3.o -o libxxx.so 

2.多个源文件/目标生成静态库
a.
ar -rc libxxx.a xxx1.o xxx2.o xxx3.o (正确方法)
b.
ar -rc libxxx.a xxx1.c xxx2.c xxx3.c (静态库可以生成;当运行连接了该静态库的可执行程序会报错:could not read symbols:Archive has no index;run ranlib to add one)

链接静态库

gcc -o main2 main.o -L./ -ladd_minus

链接动态库

  1. 在外部告诉程序,动态库在哪里
    有两种方法:
  • 将 libmulti_div.so copy到/lib/ 或 /usr/lib/ 下
    这个方法对很多软件都要使用的库比较友好
  • 在 LD_LIBRARY_PATH 变量中指定库文件路径
    这个一般就是临时弄一下。用法:
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/yourpath
  1. 编译链接
gcc -o main4 main.o -L./ -ladd_minus -lmulti_div

一、GCC 简介

《gcc命令》http://man.linuxde.net/gcc

gcc作为编译工具,用在Linux操作系统中,可以编译C、C++、Object-C、JAVA等语言。编译过程中可以带编译选项,选择编译过程。

GCC编译流程

1)预处理 Pre-Processing
2)编译 Compiling
3)汇编 Assembling
4)链接 Linking

GCC编译选项

1、一般选项

1) -c :只编译,不链接,生成目标文件“.o”。
2) -S :只编译,不汇编,生成汇编代码“.S”。
3) -E :只进行预编译/预处理,不做其他处理。

4) -o file:输出文件名为file
5) -g :在可执行程序中包含标准调试信息。
6) -v :打印出编译器内部编译各过程的命令行信息和编译器的版本。
7) -I dir :在头文件的搜索路径列表中添加dir目录
8) -L dir :在库文件的搜索路径列表中添加dir目录
9) -static :连接静态库(静态库也可以用动态库链接方式链接)
10) -llibrary :连接名为library的库文件(显示指定需要链接的动态库文件)

-shared 该选项指定生成动态连接库
l -fPIC:表示编译为位置独立的代码,不用此选项的话编译后的代码是位置相关的所以动态载入时是通过代码拷贝的方式来满足不同进程的需要,而不能达到真 正代码段共享的目的。
l -ltest:编译器查找动态连接库时有隐含的命名规则,即在给出的名字前面加上lib,后面加上.so来确定库的名称
l LD_LIBRARY_PATH:这个环境变量指示动态连接器可以装载动态库的路径。
l 当然如果有root权限的话,可以修改/etc/ld.so.conf文件,然后调用 /sbin/ldconfig来达到同样的目的,不过如果没有root权限,那么只能采用输出LD_LIBRARY_PATH的方法了。
   

2、gcc告警和出错选项

1) -ansi :支持符合ANSI标准的C程序
2) -pedantic :允许发出ANSI C标准所列出的全部警告信息
3) -pedantic-error :允许发出ANSI C标准所列出的全部错误信息
4) -w :关闭所有警告
5) -Wall :允许发出gcc提供的所有有用的报警信息
6) -werror :把所有的告警信息转化为错误信息,并在告警发生时终止编译过程


3、gcc优化选项

1)-On:n的取值范围不一致,比较典型的范围为0变化到2或者3。

虽然优化选项可以加速代码的运行速度,但对于调试而言将是一个很大的挑战。因为代码在经过优化之后,原先在源程序中声明和使用的变量很可能不再使用,控制流也可能会突然跳转到意外的地方,循环语句也可能因为循环展开而变得到处都有。
 

二、GCC生成动态库和静态库

(1)静态库
     静态是指每个用到该库的应用程序都拥有一份自己的库拷贝;
(2) 共享库
     一个共享库有可能被多个所有应用程序共享。

1)动态库生成
1.单个源文件/目标直接生成动态库
a. 
gcc -fPIC -shared xxx.c -o libxxx.so
b.
gcc -fPIC -shared xxx.o -o libxxx.so


2.多个源文件/目标生成动态库
a.
gcc -fPIC -shared xxx1.c xxx2.c xxx3.c -o libxxx.so 
b.
gcc -fPIC -shared xxx1.o xxx2.o xxx3.o -o libxxx.so 


2)静态库生成

简单地说,静态库是一个目标文件的简单集合。因此,首先有目标文件。

第一步:
gcc -c myfunc.c myproc.c将得到myfunc.o和myproc.o。

第二步:由ar(archive,归档的意思)把多个目标文件集合起来。
$ar -r libmyjob.a myfunc.o myproc.o
    

通常,静态库的命名方式应遵守libXXXXX.a格式。应用程序在使用静态库的时候,通常只需要把命名中的XXXXX部分传递给gcc即可。

1.单个源文件/目标直接生成静态库
a.
ar -rc libxxx.a xxx.o(正确方法)
b. ar -rc libxxx.a xxx.c (静态库可以生成;当运行连接了该静态库的可执行程序会报错:could not read symbols:Archive has no index;run ranlib to add one)


2.多个源文件/目标生成静态库
a.
ar -rc libxxx.a xxx1.o xxx2.o xxx3.o (正确方法)
b.
ar -rc libxxx.a xxx1.c xxx2.c xxx3.c (静态库可以生成;当运行连接了该静态库的可执行程序会报错:could not read symbols:Archive has no index;run ranlib to add one)


四、多个源文件生成一个可执行文件
gcc xxx1.c xxx2.c xxx3.c xxx4.c main.c -o main

三、生成动态库和静态库实例

gcc编译工具生成动态库和静态库之二- 

四、静态库和动态库的使用和配置

(1)静态库
     静态是指每个用到该库的应用程序都拥有一份自己的库拷贝;应用程序运行的时候,即使将库删除也没有问题,因为应用程序自己已经有了自己的拷贝。
(2) 共享库
     一个共享库有可能被多个所有应用程序共享。因此,对每个应用程序来说,即使不再使用某个共享库,也不应将其删除。此外,应用程序需要正确的环境变量设置 (LD_LIBRARY_PATH),从而找到共享库所在的位置,否则,应用程序运行时会报告找不到这个库。


二、关于使用库的问题


     对于静态库来说,gcc 的命令行参数(-I, -L)shell的环境变量(C_INCLUDE_PATH, LIBRARY_PATH)
     对于共享库来说,程序在运行时,如果用到了动态库,也需要找到对应的动态库文件;实现的方法:
shell的环境变量 (LD_LIBRARY_PATH)

 

静态库的使用

1) gcc命令行参数(-I, -L)

 默认情况下,gcc会自动搜索下面的路径:
对 头文件:
 

/usr/local/include/
/usr/include/
对库文件:
/usr/local/lib/
/usr/lib/

开发者自定义路径:
  
 例如:如果工程涉及到GDBM(GNU DataBase Management)包,需要libgdbm库,而系统中安装GDBM的路径是:
 

头文件:/opt/gdbm-1.8.3 /include
库文件:/opt/gdbm-1.8.3/lib/
  那么,gcc的命令参数是:
$gcc … -I/opt/gdbm-1.8.3/include  -L/opt/gdbm-1.8.3/lib –lgdbm

 注意:为保证兼容性,必须坚决杜绝在C/C++源文件的#include语句中或者其他相关语句中使用上述路径。

2) shell环境变量(Environmental Variable)

     除了用命令行参数,还可以用环境变量来指示gcc搜索适当的路径。
(1)Bash
     对于Bash来说,除了由系统管理员配置的内容以外,每个用户的用户目录($HOME)下,有个.bash_profile文件。可在该文件内,增加下面 的两个语句来设置GDBM头文件路径的环境变量:

C_INCLUDE_PATH=/opt/gdbm-1.8.3/include
export C_INCLUDE_PATH

     类似地,在该文件内用下面的两个语句来设置库文件路径的环境变量:

LIBRARY_PATH=/opt/gdbm-1.8.3/lib
export LIBRARY_PATH

     在.bash_profile中有了上述语句以后,就不用再使用-I和-L来搜索特定包的路径了。但是链接库的时候,还是要用-l选项。
$gcc … –lgdbm
  

   在Bash下,要检查有什么样的环境变量,可用env命令。
$env

共享库的使用

     使用共享库的应用程序,要通过环境变量LD_LIBRARY_PATH找到对应的共享库文件,不要忘了之前已经设置的共享库目录,所以是=xxxx:$LD_LIBRARY_PATH,即赋值新增的路径和原来的路径:

LD_LIBRARY_PATH=/opt/gdbm-1.8.3/lib:$LD_LIBRARY_PATH
export LD_LIBRARY_PATH

转自:

gcc -E hello.c -o hello.i   对hello.c文件进行预处理,生成了hello.i 文件
gcc -S hello.i -o hello.s    对预处理文件进行编译,生成了汇编文件
gcc -c hello.s -o hello.o  对汇编文件进行编译,生成了目标文件
gcc hello.o -o hello 对目标文件进行链接,生成可执行文件
gcc hello.c -o hello 直接编译链接成可执行目标文件
gcc -c hello.c 或 gcc -c hello.c -o hello.o 编译生成可重定位目标文件

不到你指定链接的so错误

调用动态库的时候有几个问题会经常碰到,有时,明明已经将库的头文件所在目录 通过 “-I” include进来了,库所在文件通过 “-L”参数引导,并指定了“-l”的库名,但通过ldd命令察看时,就是死活找不到你指定链接的so文件,这时你要作的就是通过修改 LD_LIBRARY_PATH或者/etc/ld.so.conf文件来指定动态库的目录。通常这样做就可以解决库无法链接的问题了。

编译链接静态库、动态库

 链接静态库

上节的1~3已经得到了libadd_minus.a和main.o文件,这一步需要把这两个文件链接起来

gcc -o main2 main.o -L./ -ladd_minus

说明1:-L./表明库文件位置在当前文件夹

说明2: -ladd_minus 表示链接 libadd_minus.a 文件,使用“-l”参数时,前缀“lib”和后缀“.a”是需要省略的。

编译链接动态库

链接动态库

  1. 在外部告诉程序,动态库在哪里
    有两种方法:
  • 将 libmulti_div.so copy到/lib/ 或 /usr/lib/ 下
    这个方法对很多软件都要使用的库比较友好
  • 在 LD_LIBRARY_PATH 变量中指定库文件路径
    这个一般就是临时弄一下。用法:
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/yourpath
  1. 编译链接
gcc -o main4 main.o -L./ -ladd_minus -lmulti_div