(22)
(3)
(2)
(17)
分类: windows
2015-05-08 09:27:17
原文地址: 作者:
将 c 函数声明为``extern "c"''(在你的 c 代码里做这个声明),然后调用它(在你的 c 或者 c 代码里调用)。例如:
// c code:
extern "c" void f(int);
void f(int i)
{
// ...
}
然后,你可以这样使用 f():
/* c code: */
void f(int);
void cc(int i)
{
f(i);
/* ... */
}
当然,这招只适用于非成员函数。如果你想要在 c 里调用成员函数(包括虚函数),则需要提供一个简单的包装(wrapper)。例如:
// c code:
class c
{
// ...
virtual double f(int);
};
extern "c" double call_c_f(c* p, int i) // wrapper function
{
return p->f(i);
}
然后,你就可以这样调用 c::f():
/* c code: */
double call_c_f(struct c* p, int i);
void ccc(struct c* p, int i)
{
double d = call_c_f(p,i);
/* ... */
}
如果你想在 c 里调用重载函数,则必须提供不同名字的包装,这样才能被 c 代码调用。例如:
// c code:
void f(int);
void f(double);
extern "c" void f_i(int i) { f(i); }
extern "c" void f_d(double d) { f(d); }
然后,你可以这样使用每个重载的 f():
/* c code: */
void f_i(int);
void f_d(double);
void cccc(int i,double d)
{
f_i(i);
f_d(d);
/* ... */
}
注意,这些技巧也适用于在 c 里调用 c 类库,即使你不能(或者不想)修改 c 头文件。
#include "aa.h"
int sample::method()
{
cout<<"method is called!\n";
}
#include
using namespace std;
class sample
{
public:
int method();
};
将上面的两个文件生成动态库libaa.so放到 /usr/lib目录下,编译命令如下
sudo g -fpic -shared -g -o /usr/lib/libaa.so aa.cxx -i ./
由于在c中不能识别类,所以要将上面类的成员函数封装成c接口函数输出,下面进行封装,将输出接口转换成c接口。
#include "add.h"
#ifndef _cplusplus
#define _cplusplus
#include "mylib.h"
#endif
int myfunc()
{
sample ss;
ss.method();
return 0;
}
#ifdef _cplusplus
extern "c"
{
#endif
int myfunc();
#ifdef _cplusplus
}
#endif
在linux下,gcc编译器并没用变量_cplusplus来区分是c代码还是c 代码,如果使用gcc编译器,这里我们可以自己定义一个变量_cplusplus用于区分c和c 代码,所以在mylib.cxx中定义了一个变量_cplusplus用于识别是否需要“extern "c"”将函数接口封装成c接口。但是如果使用g 编译器则不需要专门定义_cplusplus,编译命令如下:
g -fpic -shared -g -o mylib.so mylib.cxx -la -i ./
#include
#include
#include "mylib.h"
int
main()
{
int (*dlfunc)();
void *handle; //定义一个句柄
handle = dlopen("./mylib.so", rtld_lazy);//获得库句柄
dlfunc = dlsym(handle, "myfunc"); //获得函数入口
(*dlfunc)();
dlclose(handle);
return 0;
}
编译命令如下:
gcc -o main main.c ./mylib.so -ldl
下面就可以执行了。
需要说明的是,由于main.c 和 mylib.cxx都需要包含mylib.h,并且要将函数myfunc封装成c接口函数输出需要“extern "c"”,而c又不识别“extern "c"”,所以需要定义_cplusplus来区别处理mylib.h中的函数myfunc。
在main.c的main函数中直接调用myfunc()函数也能执行,这里介绍的是常规调用库函数的方法。
上一篇:
下一篇: