我们以上以节中的C语言写的线程池为例,包括三个文件:thread_pool.cpp、thread_pool.h、test.cpp,把线程池组件thread_pool.cpp封装为一个动态链接库或者静态链接库。
1、编译生成动态链接库(参考)
(1)编译生成.so库
g++ -w thread_pool.cpp -fPIC -shared -o libthread_pool.so
-shared:该选项指定生成动态连接库(让连接器生成T类型的导出符号表,有时候也生成弱连接W类型的导出符号),不用该标志外部程序无法连接。相当于一个可执行文件
-fPIC:表示编译为位置独立的代码,不用此选项的话编译后的代码是位置相关的所以动态载入时是通过代码拷贝的方式来满足不同进程的需要,而不能达到真正代码段共享的目的。
-w:编译时忽略警告信息
(2)将.so库链接入主文件
g++ -w test.cpp -L. -lthread_pool -o test -lpthread
-L.:表示要连接的库在当前目录中;
-lthread_pool:编译器查找动态连接库时有隐含的命名规则,即在给出的名字前面加上lib,后面加上.so来确定库的名称;
-lpthread:pthread库不是 Linux 系统默认的库,连接时需要使用动态库libpthread.so。
(3)运行
./test
报错:cannot open shared object file: No such file or directory
那就表示系統不知道xxx.so 放在哪个目录下,解决方案如下(参考):
export LD_LIBRARY_PATH=where_you_install_lib_path:$LD_LIBRARY_PATH
sudo ldconfig
(4)通过ldd可以查看链接的动态库情况:
ldd test
输出:
linux-vdso.so.1 => (0x00007ffe625dc000)
libthread_pool.so => /your_path/libthread_pool.so (0x00007f49837dd000)
libpthread.so.0 => /lib/x86_64-linux-gnu/libpthread.so.0 (0x00007f498359f000)
libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f49831d5000)
/lib64/ld-linux-x86-64.so.2 (0x0000560ad819f000)
2、编译生成静态链接库
(1)编译生成静态库.so
g++ -w -c thread_pool.cpp -o thread_pool.o #这里没有使用-shared
ar -r libthread_pool.a thread_pool.o #这里的ar相当于tar的作用,将多个目标打包。
(2)把静态库链接入主文件
g++ -w test.cpp -lthread_pool -L. -static -o test -lpthread
-static:选项是告诉编译器,-lthread_pool是静态库,如果不指定-static,则既可以链接动态库也可以链接静态库,指定了-static之后只能链接静态库。
或者:
g++ -w test.cpp libthread_pool.a -L. -o test -lpthread
(3)运行
./test
静态链接生成的可执行文件不能使用ldd命令。此外,使用静态库链接链接的文件运行不会出现动态库的找不到文件的问题: cannot open shared object file: No such file or directory,因此不需要额外指定库目录。
参考:https://blog.youkuaiyun.com/a600423444/article/details/7206015
本文介绍如何将C语言编写的线程池组件thread_pool.cpp封装成动态链接库(.so)或静态链接库(.a)。包括编译生成库文件的具体命令及参数解释,以及如何将这些库文件链接到主程序中。
1357

被折叠的 条评论
为什么被折叠?



