已验证!
做个 搞清摄像机时要用到他的动态库.so。参考此文学习了一下动态库。验证过程,和此文有异同的地方我会在()里注明。 蛮清楚的。 我们通常把一些公用函数制作成函数库,供其它程序使用。函数库分为静态库和动态库两 在创建函数库前,我们先来准备举例用的源程序,并将函数库的源程序编译成.o文件。
hello.c(见程序2)是函数库的源程序,其中包含公用函数hello,该函数将在屏幕上输出" #ifndef HELLO_H void hello(const char *name); #endif //HELLO_H #include <stdio.h> void hello(const char *name) #include "hello.h" int main() 第2步:将hello.c编译成.o文件; 无论静态库,还是动态库,都是由.o文件创建的。因此,我们必须将源程序hello.c通过g 在系统提示符下键入以下命令得到hello.o文件。 # gcc -c hello.c # 我们运行ls命令看看是否生存了hello.o文件。 # ls hello.c hello.h hello.o main.c #
下面我们先来看看如何创建静态库,以及使用它。 第3步:由.o文件创建静态库; 静态库文件名的命名规范是以lib为前缀,紧接着跟静态库名,扩展名为.a。例如:我们将 在系统提示符下键入以下命令将创建静态库文件libmyhello.a。 # ar crv libmyhello.a hello.o # 我们同样运行ls命令查看结果: # ls hello.c hello.h hello.o libmyhello.a main.c # ls命令结果中有libmyhello.a。 第4步:在程序中使用静态库; 静态库制作完了,如何使用它内部的函数呢?只需要在使用到这些公用函数的源程序中包 在程序3:main.c中,我们包含了静态库的头文件hello.h,然后在主程序main中直接调用公 (# gcc -o hello main.c -L. -lmyhello??) (这句是作者的注释可以不用理) #gcc main.c libmyhello.a -o main # ./hello Hello everyone! (这里估计是作者写错了,是# ./main) # 我们删除静态库文件试试公用函数hello是否真的连接到目标文件 hello中了。 # rm libmyhello.a rm: remove regular file `libmyhello.a'? y # ./hello (这里估计是作者写错了,是# ./main) Hello everyone! # 程序照常运行,静态库中的公用函数已经连接到目标文件中了。 我们继续看看如何在Linux中创建动态库。我们还是从.o文件开始。 第5步:由.o文件创建动态库文件; 动态库文件名命名规范和静态库文件名命名规范类似,也是在动态库名增加前缀lib,但其 在系统提示符下键入以下命令得到动态库文件libmyhello.so。 # gcc -shared -fPCI -o libmyhello.so hello.o # 我们照样使用ls命令看看动态库文件是否生成。 # ls hello.c hello.h hello.o libmyhello.so main.c # 第6步:在程序中使用动态库; 在程序中使用动态库和使用静态库完全一样,也是在使用到这些公用函数的源程序中包含 # gcc -o hello main.c -L. -lmyhello # ./hello ./hello: error while loading shared libraries: libmyhello.so: cannot open shar # 哦!出错了。快看看错误提示,原来是找不到动态库文件libmyhello.so。程序在运行时, # mv libmyhello.so /usr/lib (这一步由于我用的不是root用户,移动时提示权限不够,在这步前加条$su命令,取权限) # ./hello Hello everyone! # ( 这步后我没有成功,无论是用#权限还是$都报错说我权限不够。报错内容如下: [root@localhost 20090505]# ./hello ./hello: error while loading shared libraries: /usr/lib/libmyhello.so: cannot restore segment prot after reloc: Permission denied [root@localhost 20090505]# exit exit [pin@localhost 20090505]$ ./hello ./hello: error while loading shared libraries: /usr/lib/libmyhello.so: cannot restore segment prot after reloc: Permission denied 查了一下发现是原来这是 SELinux搞的鬼,解决办法有两个 1. 使用chcon 命令 示例: chcon -t texrel_shlib_t /usr/local/rsi/idl_6.1/bin/bin.linux.x86/*.so (这步我从 SELinux故障诊断浏览器的提示,用了chcon -t texrel_shlib_t /usr/lib/libmyhello.so命 令) 2.禁止掉selinux 更改/etc/sysconfig/selinux 文件的内容为 SELINUX=disabled ) 成功了。这也进一步说明了动态库在程序运行时是需要的。 # ldd hello执行 test,可以看到它是如何调用动态库中的函数的。 [pin@localhost 20090505]$ ldd hello linux-gate.so.1 => (0x00110000) libmyhello.so => /usr/lib/libmyhello.so (0x00111000) libc.so.6 => /lib/libc.so.6 (0x00859000) /lib/ld-linux.so.2 (0x0083a000) 我们回过头看看,发现使用静态库和使用动态库编译成目标程序使用的gcc命令完全一样, 先删除除.c和.h外的所有文件,恢复成我们刚刚编辑完举例程序状态。 # rm -f hello hello.o /usr/lib/libmyhello.so # ls hello.c hello.h main.c # 在来创建静态库文件libmyhello.a和动态库文件libmyhello.so。 # gcc -c hello.c # ar cr libmyhello.a hello.o # gcc -shared -fPCI -o libmyhello.so hello.o # ls hello.c hello.h hello.o libmyhello.a libmyhello.so main.c # 通过上述最后一条ls命令,可以发现静态库文件libmyhello.a和动态库文件libmyhello.s # gcc -o hello main.c -L. -lmyhello # ./hello ./hello: error while loading shared libraries: libmyhello.so: cannot open shar # 从程序hello运行的结果中很容易知道,当静态库和动态库同名时, gcc命令将优先使用动 Note: 调用动态库的时候有几个问题会经常碰到,有时,明明已经将库的头文件所在目录 通过 “-I” include进来了,库所在文件通过 “-L”参数引导,并指定了“-l”的库名,但通过ldd命令察看时,就是死活找不到你指定链接的so文件,这时你要作的就是通过修改 LD_LIBRARY_PATH或者/etc/ld.so.conf文件来指定动态库的目录。通常这样做就可以解决库无法链接的问题了。 静态库链接时搜索路径顺序: |
linux下用gcc生成静态库和动态库
最新推荐文章于 2024-11-14 09:14:03 发布
linux下用gcc生成静态库和动态库
2010年08月10日 星期二 上午 11:59