如果你有一个c++做的动态链接库.so文件,而你只有一些相关类的声明, 那么你如何用c调用呢,别着急,本文通过一个小小的例子,让你能够很爽的搞定.
一、编译生成一个供测试使用的C++动态库:
链接库头文件:
//head.h
class A
{
public:
A();
virtual ~A();
int gt();
int pt();
private:
int s;
};
//firstso.cpp
#include <iostream>
#include "head.h"
A::A(){}
A::~A(){}
int A::gt()
{
s=10;
}
int A::pt()
{
std::cout<<s<<std::endl;
}
动态库编译命令如下:
g++ -shared -o libmy.so firstso.cpp
有时会报错并提示加上-fPIC参数,加上即可:
g++ -fPIC -shared -o libmy.so firstso.cpp
这时候生成libmy.so文件,将其拷贝到系统库里面:/usr/lib/或配置到LD_LIBRARY_PATH
或者
生成静态库,静态库编译命令如下:
g++ -c firstso.cpp
ar -rc libmy.a libmy.so
这时候生成libmy.a文件,待下一步使用。
二、对C++动态库里的接口进行二次封装:
//secso.cpp
#include <iostream>
#include "head.h"
extern "C"
{
int f();
int f()
{
A a;
a.gt();
a.pt();
return 0;
}
}
编译命令:
使用动态库:
gcc -shared -o sec.so secso.cpp -L. -lmy
或者
使用静态库:
gcc -shared -o sec.so secso.cpp libmy.a
这时候生成第二个.so文件,此时库从一个类变成了一个c的接口.
备注:
在使用嵌入式编译器移植的时候,使用了xxxx-gcc编译出来的第二个.so文件,移植后dlopen会出现
undefined symbol: _ZTVN10_cxxabiv120__si_class_type_infoE
的错误。将xxxx-gcc改为xxxx-g++生成的第二个.so文件,移植后则能正常运行。
C程序调用接口:
下面开始调用:
//test.c
#include "stdio.h"
#include "dlfcn.h"
#define SOFILE "sec.so"
int (*f)();
int main()
{
void *dp;
dp=dlopen(SOFILE,RTLD_LAZY);
f=dlsym(dp,"f");
f();
return 0;
}
编译命令如下:
运行
先将libmy.so加入到环境变量LD_LIBRARY_PATH:
如libmy.so在tmp目录
export LD_LIBRARY_PATH=/tmp:$LD_LIBRARY_PATH
./myapp
10
Linux 中C文件调用C++代码的方法(静态库)
如何一个C语言的工程中调用C++写成的代码呢?最简单的方法当然是使用g++编译C工程,但有时候可能因为一些限制不能使用g++编译,这时候可以考虑把C++代码Make成一个静态库,再在C代码中调用。具体做法如下:
假设有一个类文件如下:
//MyClass.cpp
class MyClass
{
public:
int add(int a, int b)
{
return (a + b);
}
};
我们首先将该类的方法进行包装,形成类似C语言的库函数。
//output c function
extern "C" int test_add(int a, int b);
void test_add(int a, int b)
{
MyClass h;
return h.add(a, b);
}
使用如下命令对上面的MyClass.cpp进行编译,并生成归档文件:
g++ -c MyClass.cpp
ar crv libMyClass.a MyClass.o
在C语言中调用该函数方法:
//test.c
#include <stdio.h>
extern int test_add(int a,int b);
int main()
{
printf("2 + 4 = %d\n", test_add(2, 4));
return 0;
}
使用如下命令编译该文件:
gcc -o test test.c libMyClass.a -lstdc++