C语言中没有类的概念,如何调用呢,需要封装一下,增加一个中间层。这个中间层屏蔽了类,提供一个函数给上层,并且还要用c++编译器来编译。
下面举例,说明一个封装策略:
- //code in add.cxx
- #include "add.h"
- int sample::method()
- {
- cout<<"method is called!\n";
- }
- //code in add.h
- #include <iostream>
- using namespace std;
- class sample
- {
- public:
- int method();
- };
将上面的两个文件生成动态库libadd.so放到 /usr/lib目录下,编译命令如下:
sudo g++ -fpic -shared -g -o /usr/lib/libadd.so add.cxx -I ./
由于在C中不能识别类,所以要将上面类的成员函数,要封装成C接口函数才能被调用。下面进行封装,将输出接口转换成C接口。
- //code in mylib.cxx
- #include "add.h"
- #ifndef _cplusplus
- #define _cplusplus
- #include "mylib.h"
- #endif
- int myfunc()
- {
- sample ss;
- ss.method();
- return 0;
- }
- //code in mylib.h
- #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 ./
- main.c
- #include <stdio.h>
- #include <dlfcn.h>
- #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()函数也能执行,这里介绍的是常规调用库函数的方法。