Linux C++简单生成与调用so库

Linux C++简单生成与调用so库


网上有很多C语言编译so库的方法,用C++的偏少,遇见了一些坑,这里记录一下,以作参照。

生成so库

先实现一个最简单的函数,libtest.cpp(先不考虑extern "C"这句话):

#include<iostream>
//extern "C"
double ddd(double a){
	return a*2;
}

linux 命令(注意这里生成动态库文件时要加上-fPIC -shared):

[root@localhost jay]# g++ -fPIC -shared libtest.cpp -o libtest.so 

至此,so库文件完成。

调用so库

初探

main.cpp(这是最常见,最自然的写法):

#include<iostream>
#include<dlfcn.h>
//extern "C"
int main(){
	void *handle = dlopen("./libtest.so",RTLD_LAZY);
	double (*ans) (double);
	char *error;
	ans = (double(*)(double))dlsym(handle,"ddd");
	//ans = (double(*)(double))dlsym(handle,"_Z3dddd");
    std::cout<<(*ans)(2.0)<<std::endl;
	dlclose(handle);
	return 0;
}

Linux命令:

[root@localhost jay]# g++ -o a.out main.cpp -ldl

可以成功执行,然而:

[root@localhost jay]# ./a.out 
段错误

解决

因为发现别人用C语言都是可以正常编译的,找到wwxxlld大佬的这篇文章,利用ldd命令发现没有依赖缺失情况,使用nm命令:

发现是g++编译器会将函数名进行优化,以便实现c++的重载。所以最简单的方法就是给外部使用的函数需要加上声明,extern “C” 函数名 ,让编译器编译时以c语言的方式去编译,不要优化该函数名,或者直接全局声明 extern “C”
在这里插入图片描述
当然,你也可以直接调用最开始nm得到的g++优化后的新函数名 _Z3dddd……(虽然似乎有点蠢)。
不管怎样,现在问题都能解决了:

[root@localhost jay]# g++ -o a.out main.cpp -ldl
[root@localhost jay]# ./a.out 
4

参考资料

[1]: Linux 动态加载并调用动态库(.so)方法介绍
[2]: linux中的nm命令简介
[3]: 程序调用动态库,编译通过,调用动态库函数运行出现undefined symbol错误

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、付费专栏及课程。

余额充值