自己制作静态和动态链接库并使用

来源:互联网 发布:java short类型 编辑:程序博客网 时间:2024/06/04 19:47
自己制作静态链接库并使用
(1)第一步:自己制作静态链接库
    首先使用gcc -c只编译不连接,生成.o文件;然后使用ar工具进行打包成.a归档文件
    库名不能随便乱起,一般是lib+库名称,后缀名是.a表示是一个归档文件
注意:制作出来了静态库之后,发布时需要发布.a文件和.h文件。
    gcc aston.c -o aston.o -c
    ar -rc libaston.a aston.o

(2)第二步:使用静态链接库
    把.a和.h都放在我引用的文件夹下,然后在.c文件中包含库的.h,然后直接使用库函数。
第一次,编译方法:gcc test.c -o test
报错信息:test.c:(.text+0xa): undefined reference to `func1'

test.c:(.text+0x1e): undefined reference to `func2'
第二次,编译方法:gcc test.c -o test -laston
报错信息:/usr/bin/ld: cannot find -laston
collect2: error: ld returned 1 exit status
第三次,编译方法:gcc test.c -o test -laston -L.
无报错,生成test,执行正确。
(3)除了ar名另外,还有个nm命令也很有用,它可以用来查看一个.a文件中都有哪些符号


自己制作动态链接库并使用
(1)动态链接库的后缀名是.so(对应windows系统中的dll),静态库的扩展名是.a
(2)第一步:创建一个动态链接库。
    gcc aston.c -o aston.o -c -fPIC      //-fPIC位置无关码
    gcc -o libaston.so aston.o -shared 
        -fPIC是位置无关码,-shared是按照共享库的方式来链接。
注意:做库的人给用库的人发布库时,发布libxxx.so和xxx.h即可。
(3)第二步:使用自己创建的共享库。
第一步,编译方法:gcc test.c -o test
报错信息:test.c:(.text+0xa): undefined reference to `func1'

test.c:(.text+0x1e): undefined reference to `func2'
collect2: error: ld returned 1 exit status

第二步,编译方法:gcc test.c -o test -laston
报错信息:/usr/bin/ld: cannot find -laston
collect2: error: ld returned 1 exit status

第三步,编译方法:gcc test.c -o test -laston -L.
编译成功

但是运行出错,报错信息:
error while loading shared libraries: libaston.so: cannot open shared object file: No such file or directory

错误原因:动态链接库运行时需要被加载(运行时环境在执行test程序的时候发现他动态链接了libaston.so,于是乎会去固定目录尝试加载libaston.so,如果加载失败则会打印以上错误信息。)

解决方法一:
将libaston.so放到固定目录下就可以了,这个固定目录一般是/usr/lib目录。
cp libaston.so /usr/lib即可

解决方法二:使用环境变量LD_LIBRARY_PATH。操作系统在加载固定目录/usr/lib之前,会先去LD_LIBRARY_PATH这个环境变量所指定的目录下去寻找,如果找到就不用去/usr/lib下面找了,如果没找到再去/usr/lib下面找。所以解决方案就是将libaston.so所在的目录导出到环境变量LD_LIBRARY_PATH中即可。
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/mnt/hgfs/Winshare/s5pv210/AdvancedC/4.6.PreprocessFunction/4.6.12.sharedobject.c/sotest

在ubuntu中还有个解决方案三,用ldconfig

(4)ldd命令:作用是可以在一个使用了共享库的程序执行之前解析出这个程序使用了哪些共享库,并且查看这些共享库是否能被找到,能被解析(决定这个程序是否能正确执行)。


一步到位的编译指令是:gcc test.c -o test
预处理:gcc -E test.c -o test.i 或 gcc -E test.c                 //gcc的-E选项,可以让编译器在预处理后停止,并输出预处理结果。
预处理之后,可直接对生成的test.i文件编译,生成汇编代码:gcc -S test.i -o test.s     //gcc的-S选项,表示在程序编译期间,在生成汇编代码后,停止,-o输出汇编代码文件。
对于上一小节中生成的汇编代码文件test.s,gas汇编器负责将其编译为目标文件:gcc -c test.s -o test.o
对于上一小节中生成的test.o,将其与C标准输入输出库进行连接,最终生成程序test:gcc test.o -o test    //gcc连接器是gas提供的,负责将程序的目标文件与所需的所有附加的目标文件连接起来,最终生成可执行文件。附加的目标文件包括静态连接库和动态连接库。
执行:./test
假设有一个由test1.c和 test2.c两个源文件组成的程序,为了对它们进行编译,并最终生成可执行程序test:gcc test1.c test2.c -o test
编译成目标文件(.i):gcc –c –I /usr/dev/mysql/include test.c –o test.o
把所有目标文件链接成可执行文件:gcc –L /usr/dev/mysql/lib –lmysqlclient test.o –o test//Linux下的库文件分为两大类分别是动态链接库(通常以.so结尾)和静态链接库(通常以.a结尾),二者的区别仅在于程序执行时所需的代码是在运行时动态加载的,还是在编译时静态加载的。
1 0
原创粉丝点击