前言
上一篇文章简单介绍了*NIX下的动态库的使用,我们在这篇文章中实现一个计算器,计算器程序calc本身不做运算,只是将操作数传递给具体的插件(adder, suber, muler, diver)来完成实际运算。首先,计算器根据插件配置文件plugin.xml来确定插件的位置,名称,入口符号的定义,然后依次调用各个插件完成计算。
插件列表
文中涉及到的插件定义在plugin.xml中,文档结构如下:
<plugins>
<plugin name="adder">
<library path="/home/juntao/.libs/adder.so">
</library>
<entry name="add">
</entry>
</plugin>
<plugin name="suber">
<library path="/home/juntao/.libs/suber.so">
</library>
<entry name="sub">
</entry>
</plugin>
<plugin name="muler">
<library path="/home/juntao/.libs/muler.so">
</library>
<entry name="mul">
</entry>
</plugin>
<plugin name="diver">
<library path="/home/juntao/.libs/diver.so">
</library>
<entry name="div">
</entry>
</plugin>
</plugins>
每个插件为一个plugin标签,plugin标签中包含library, entry两个字标签,分别定义动态库文件的路径及名称和插件函数的入口。为了简便,我们不重复设计list及xml解析,这里使用libxml2作为xml的分析器,GLIB中的GSList(单链表)来作为插件列表的链表对象。
每一个插件在C语言中的定义如下,非常简单(plugin.h)
#ifndef _PLUGIN_H_
#define _PLUGIN_H_
typedef struct{
char name[64];
char path[256];
char entry[128];
int version;
}Plugin;
#endif
这里为了行文方便,Plugin结构中的字符串为静态尺寸。
计算器
计算器调用parseconf模块中的load_plugins将plugin.xml中定义的Plugin加载进一个GSList,以备后用:
#include "plugin.h"
extern int load_plugins(char *config, GSList **list);
插件中的函数原型应该符合接口定义:
//pointer to function, which return a double, and get 2 double as input
double (*pfunc)(double a, double b);
计算器的主要代码如下:
int calc_test(double a, double b){
GSList *list = NULL, *it = NULL;
Plugin *pl = NULL;
//insert a null node into list at first
list = g_slist_append(list, NULL);
int code = 0;
double result;
//load plugin defined in plugin.xml into list
load_plugins("plugin.xml", &list);
for(it = list; it != NULL; it = it->next){
pl = (Plugin *)it->data;
if(pl == NULL){
continue;
}else{
//open the library
flib = dlopen(pl->path, RTLD_LAZY);
dlError = dlerror();
if(dlError){
fprintf(stderr, "open %s failed\n", pl->name);
g_slist_free(list);
return -1;
}
//get the entry
*(void **)(&pfunc) = dlsym(flib, pl->entry);
dlError = dlerror();
if(dlError){
fprintf(stderr, "find symbol %s failed\n", pl->entry);
g_slist_free(list);
return -1;
}
//call the function
result = (*pfunc)(a, b);
printf("%s(%f, %f) = %f\n", pl->entry, a, b, result);
//then close it
code = dlclose(flib);
dlError = dlerror();
if(code){
fprintf(stderr, "close lib error\n");
g_slist_free(list);
return -1;
}
}
}
g_slist_free(list);
return 0;
}
首先,定义一个GSList,然后将其传递给load_plugins,load_plugins解析plugin.xml,然后填充list返回,calc_test遍历插件列表,并调用每一个插件定义的entry.
除法器
我们来看一个具体的插件:做除法的模块
#include <stdio.h>
double div(double a, double b){
if(b == 0){
fprintf(stderr, "div zero error\n");
return -1;
}else{
return a / b;
}
}
diver.c在编译之后,生成diver.so,将其置于plugin.xml定义的位置处即可。
运行结果如下图所示:
其他代码如xml的解析,GSList的使用等与插件机制关系不大,感兴趣的朋友可以在附件中查看。
- 大小: 14.5 KB
分享到:
相关推荐
在本篇中,我们将深入探讨"C语言插件机制(下)"这一主题,主要关注如何在C语言中实现插件系统,以及与之相关的工具和技术。首先,插件机制是一种允许程序在运行时动态加载和卸载附加功能的方式,它极大地提高了软件的...
C语言插件通常是指在开发环境中为了增强对C语言支持而设计的工具或组件。这些插件可能提供语法高亮、自动补全、调试功能、代码分析等特性,以提升C程序员的开发效率和代码质量。 在C语言的开发过程中,一些关键知识...
本文将深入探讨如何利用CDT插件在Eclipse中构建C语言的编译环境,以及相关辅助工具和配置技巧。 首先,CDT插件提供了代码编辑、调试、构建系统等一站式服务。安装CDT后,用户可以在Eclipse内直接编写、编译和运行C/...
Sublime Text 3 是一款功能强大且广受欢迎的代码编辑器,它的插件机制使得用户可以根据需要安装各种插件以满足不同的需求。本文将详细介绍 Sublime Text 3 插件的安装方法,并以 Emmet 插件为例展示插件的安装和...
4.3 异常处理:使用`try-catch`机制或自定义错误处理流程,确保程序在异常情况下能正常退出。 五、模块化与可扩展性 5.1 封装原则:将功能相近的代码组织在一起,提高代码复用性。每个源文件只包含一个主要功能。 ...
在IT领域,尤其是在多媒体处理和流媒体应用中,GStreamer是一个强大的开源框架,它允许开发者构建复杂的...通过深入研究源代码和GStreamer插件机制,开发者可以进一步定制和优化这个插件,以满足特定应用场景的需求。
7. **动态库和插件机制**:Snort允许用户通过动态加载库(插件)来扩展其功能,比如添加新的检测规则、日志格式或协议解析器。 8. **性能优化**:为了处理海量网络流量,Snort使用了多种优化技术,如多线程处理、...
C语言本身并不直接支持面向对象编程,但通过巧妙的设计,可以模拟出类似的功能,构建出强大的框架系统和插件机制。本资源"book_cprogramming"可能是一个教程或代码示例集,旨在帮助开发者理解和掌握这一技术。 一、...
“软件/插件”可能暗示着这不只是一个简单的程序,它可能包含了一些特定的库或者插件,如EasyX,以增强其功能。“范文/模板/素材”则表明这个项目可以作为一个学习的样本,或者是开发者创建类似游戏时可以参考的模板...
5. **内存管理**:C语言没有自动垃圾回收机制,所以开发者需要手动分配和释放内存。在创建和删除蛇的节点时,要特别注意内存泄漏问题。 6. **屏幕绘图**:在C语言中,可以使用ANSI转义码或者`ncurses`库来在终端上...
2. **DLL项目设置**:在C语言环境下,如Visual Studio,你需要创建一个Win32 DLL项目。在项目属性中,配置生成动态链接库的目标类型,并确定所需的API和头文件。 3. **导出函数**:在C语言中,使用`__declspec(dll...
5. 并发控制:尽管这是一个单用户系统,但源码可能包含了一些并发控制的概念,如锁或其他同步机制,以模拟多用户环境下可能遇到的问题。 6. 日志记录:可能包含简单的日志记录功能,用于追踪系统的运行状态和用户...
创建两个线程分别控制两条蛇,通过同步机制(如互斥锁或条件变量)保证游戏逻辑的正确性,防止并发冲突。 【游戏逻辑与算法】 贪吃蛇游戏的逻辑包括蛇的移动、食物生成、碰撞检测和游戏结束条件判断。其中,移动...
现在,尽管宽带连接更为常见,但PPP在某些场景下仍然有用,例如在路由器配置、远程访问服务器和物联网设备中。 首先,让我们深入了解PPP协议。PPP协议主要由三部分组成:链路控制协议(LCP)、网络控制协议(NCP)...
10. 可扩展性:可扩展性是日历系统的重要特征之一,需要使用C语言实现可扩展机制,包括插件机制、接口机制等。 11. 可维护性:可维护性是日历系统的重要特征之一,需要使用C语言实现可维护机制,包括日志机制、错误...
学习PLL的C语言实现和仿真,有助于初学者理解PLL的工作机制,提高分析和设计能力。通过编程实现,可以更直观地看到PLL如何响应不同输入信号,以及在不同参数设置下的行为,这对于理解和优化PLL系统至关重要。 5. ...
8. 用户界面:虽然C语言本身不支持图形用户界面(GUI),但在命令行环境下,可以设计简单的文本交互界面,通过获取用户输入的命令来执行相应的功能。 9. 测试与调试:完成编码后,需要对系统进行详尽的测试,确保...
标签为"软件/插件 c语言 游戏",这进一步明确了我们要探讨的内容。C语言在游戏开发中通常用于编写底层逻辑和高性能部分,而“软件/插件”则可能指的是游戏的可扩展性和模块化设计。 在压缩包内,我们找到了名为“纯...
4. 错误处理:C语言提供了丰富的错误处理机制,如设置errno变量、使用异常处理函数等,确保在遇到问题时,系统能够给出适当的反馈,而不是崩溃。 5. 文件操作:图书信息通常存储在文件中,C语言的文件操作函数(如...
8. **错误处理**:良好的错误处理机制能确保游戏在遇到问题时不会崩溃。这通常通过设置错误检查和异常处理来实现。 9. **算法与数据结构**:游戏中的很多问题可以转化为经典的算法问题,如搜索路径、优化资源分配等...