linux c/c++ 编译加载动态库so

linux c/c++ 编译加载动态库so

创建

gcc test.c -fPIC -shared -o libtest.so

加载

GCC编译程序时,由于GCC命令不经能够编译,也能够链接程序,GCC链接程序是通过ld命令实现的,如何将GCC的命令行参数传递给ld命令呢,这就是通过-Wl,来实现的。

格式如下:

gcc -Wl,param1,param2,param3,…

在调用ld命令时,会等效于如下:

ld param1 param2 param3

注意:*-Wl,* 后的逗号(*,*)必不可少,如果要传递多个参数,参数间用*,**分隔***

-Wl,-Bstatic

指定后续ld在处理 -l 参数来链接库文件的链接方式首选静态库

-Wl,-Bdynamic

指定后续ld在处理 -l 参数来链接库文件的链接方式首选动态库

例如: -Wl,-Bstatic -la -lb -lc -Wl,-Bdynamic -ld -l e

会被解释成 :ld liba.a libb.a libc.a libd.so libe.so

-Wl,-rpath -Wl,/data/workroom/libs/lib

会被解释成:ld -rpath /data/workroom/libs/lib

# ldd XXX 命令查看依赖库
$ ldd ./out/b.c.out
root@0187031113b5:# ldd ./out/b.c.out
        linux-vdso.so.1 (0x00007fffaa8d4000)
        libtaintcpp.so => /home/workhome/build/libtaintcpp.so (0x00007fb0c59bd000)
        libpthread.so.0 => /lib/x86_64-linux-gnu/libpthread.so.0 (0x00007fb0c598b000)
        libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007fb0c5799000)
        libstdc++.so.6 => /lib/x86_64-linux-gnu/libstdc++.so.6 (0x00007fb0c55b7000)
        libm.so.6 => /lib/x86_64-linux-gnu/libm.so.6 (0x00007fb0c5468000)
        libgcc_s.so.1 => /lib/x86_64-linux-gnu/libgcc_s.so.1 (0x00007fb0c544d000)
        /lib64/ld-linux-x86-64.so.2 (0x00007fb0c59c6000)

法一:

# $(LIB_PATH): libtest.so 所在路径
# -Wl,-rpaht $(LIB_PATH): 告诉编译器运行时加载指定动态库链接
# -L$(LIB_PATH) 编译器编译时加载指定路径动态库
# -ltest 加载 libtest.so
$ gcc a.c -Wl,-rpath $(LIB_PATH) -L$(LIB_PATH) -ltest -o a.out

若无 -Wl,-rpaht $(LIB_PATH) 但在运行时,会报错
在这里插入图片描述

若只有-Wl,-rpaht $(LIB_PATH), 无-L$(LIB_PATH), 则找不到动态库

在这里插入图片描述

法二: 复制到/usr/lib

编译时添加参数 -lxxx 即可;但不推荐

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Linux下调用动态库主要是通过dlopen、dlsym、dlclose等函数来实现。下面是一个简单的示例: 1. 编写动态库代码 首先,我们需要编写一个动态库的代码。这里我们以一个简单的例子来说明,假设我们需要编写一个名为libmylib.so的动态库,其中包含一个用于加法运算的函数add。 // mylib.h #ifndef MYLIB_H #define MYLIB_H #ifdef __cplusplus extern "C" { #endif int add(int a, int b); #ifdef __cplusplus } #endif #endif //MYLIB_H // mylib.cpp #include "mylib.h" int add(int a, int b) { return a + b; } 注意,在动态库中需要将函数声明为extern "C",这是因为在C++中函数名称会被编译器进行名称重整,而在动态库中需要使用原始的函数名称,所以需要使用extern "C"来告诉编译器不要进行名称重整。 2. 编译生成动态库 编译生成动态库的命令如下: g++ -shared -fPIC -o libmylib.so mylib.cpp 其中,-shared选项表示生成动态库,-fPIC选项表示编译时需要生成位置无关代码,-o选项表示指定输出文件名。 3. 编写调用动态库的代码 我们可以在另一个C++程序中调用上述动态库。下面是一个简单的示例: // main.cpp #include <iostream> #include <dlfcn.h> #include "mylib.h" int main() { void *handle = dlopen("./libmylib.so", RTLD_LAZY); if (!handle) { std::cerr << "Cannot open library: " << dlerror() << '\n'; return 1; } typedef int (*add_t)(int, int); add_t add_func = reinterpret_cast<add_t>(dlsym(handle, "add")); if (!add_func) { std::cerr << "Cannot load symbol add: " << dlerror() << '\n'; dlclose(handle); return 1; } int result = add_func(2, 3); std::cout << "The result is " << result << '\n'; dlclose(handle); return 0; } 该程序首先通过dlopen函数打开动态库,并将返回的句柄保存在变量handle中。然后通过dlsym函数获取动态库中的add函数地址,并将其转换为函数指针类型add_t。最后,我们可以通过add_func指针调用add函数。 注意,在调用dlsym函数时需要指定原始的函数名称,即在编写动态库代码时使用的名称。 4. 编译生成可执行程序 编译生成可执行程序的命令如下: g++ -o main main.cpp -ldl 其中,-ldl选项表示链接动态库加载器库。 5. 运行程序 运行可执行程序的命令如下: ./main 程序的输出应该是: The result is 5 至此,我们成功地调用了动态库中的函数。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值