`

C++ Plug-in 技术的一些深度思考(2)

阅读更多

C++ Plug-in 技术的一些深度思考(1)

 

我们可以使用动态连接库的技术实现上诉的功能。大方向说,将CalcAdd和CalcMul编译连接成动态连接库,CalcFactory.cpp动态的加载它们。这样的话,可以实现两个功能:

[1] 如果更新libCalcAdd.so或者增加一个libCalcMin.so,不需要重新编译,链接,发布主程序calc,只需要重新发布你更新或者增加的so文件

[2] 如果我们可以横向的定义出主程序的测试规范,我们甚至不需要回归测试主程序calc。

我们重新设计了文件目录的组织结构,如下图所示:



 

有修改的文件有CalcFactory.cpp / CalcFactory.h。主要是增加了load 动态连接库的内容。 CalcAdd.cpp和CalcMul.cpp分别增加了一个C函数,返回一个动态分配的CalcAdd或者CalcMul。

// core/src/CalcFactory.cpp
#include <dlfcn.h> //for dlopen and dlsym
#include <iostream>

#include "../include/CalcFactory.h"

using std::cerr;
using std::endl;
using std::string;
using std::map;


map<string, CalcFactory::GetNewInstanceFun> CalcFactory::calcHandlers;

ICalc* CalcFactory::getCalc(string calcType)
{
    if (calcHandlers.end() == calcHandlers.find(calcType)) {
        string libName = "../lib/libCalc" + calcType + ".so";
        void *pdlHandle = dlopen(libName.c_str(), RTLD_NOW);
        if(!pdlHandle)
        {
            cerr << "Fail to load " << libName << endl;
            return 0;
        }
        char *err = dlerror();
        if(err)
        {
            cerr << "Fail to load " << libName << " due to " << err << endl;
            return 0;
        }
        void *rst = dlsym(pdlHandle, "getNewInstance");
        if (!rst) {
            cerr << "There is no \"getNewInstance\" function in " << libName << endl;
            return 0;
        }
        GetNewInstanceFun newFun = (GetNewInstanceFun)rst;
        if (!newFun) {
            cerr << "Fail to cast to (ICalc *)(*)(). " << endl;
            return 0;
        }
        err = dlerror();
        if(err)
        {
            cerr << "There is no \"getNewInstance\" function in " << libName
                 << " due to " << err <<endl;
            return 0;
        }
        calcHandlers[calcType] = newFun;
    }
    GetNewInstanceFun nf = calcHandlers[calcType];
    return nf();
}

 

注意这里的修改。

[1] 取消了这个cpp对CalcAdd.h和CalcMul.h的依赖。本质上,在Core这边里对Plug-in的使用都是完全基于接口ICalc的。

[2] 增加了一个全局的map,用来避免反复的load动态连接库。

[3] 在Unix动态连接的思路就是dlopen和dlsym,具体的技术大家可以google一下。如果有Windows的经验,可以发现这些过程Windows和Unix几乎是一致的。

 

// plugin/src/CalcAdd.cpp

#include "../include/CalcAdd.h"

void CalcAdd::calc(int *p, size_t size, int *result)
{
    for (size_t i = 0 ; i < size; ++i)
    {
        (*result) += p[i];
    }
}

CalcAdd::~CalcAdd() { }

extern "C"
{
    ICalc *getNewInstance()
    {
        return new CalcAdd;
    }
}

 

注意了,其实整个类CalcAdd.cpp的实现没有任何的改变。改变的是增加了一个函数ICalc *getNewInstance()。这个函数要注意几点:

[1] extern “C” 在这里不能省略。也许你运气好,在你的平台上省略这个东西能顺利运行通过。但是这种代码绝对不能移植,至少在我的Mac上面,不能省略这个extern "C"。本质原因是因为这个函数要被动态加载,而dlsym几乎要返回的那个函数指针是C链接的;

[2] getNewInstance()这个函数的签名,包括返回类型ICalc*, 参数个数和类型,函数名getNewInstance是不能乱改的。首先,函数名getNewInstance被CalcFactory.cpp的dlsym用作找到函数指针的索引,void *rst = dlsym(pdlHandle, "getNewInstance"); 如果函数名错了,那么这个返回的rst是NULL。如果函数签名不对,那么GetNewInstanceFun newFun = (GetNewInstanceFun)rst; 会出错。而且这种错误不容易检查,也许newFun并不是NULL。所以一旦这个Core定下来了,所有的Plug-in里面都必须是这样的函数:

extern "C"
{
    ICalc *getNewInstance()
    {
        // do something

    }
}
当然不用太担心,如果你写错了这个函数,你一测试就能发现。

其实,还有的修改就是Makefile。这个不同的平台,不同的编译器选项差别很大。我大致介绍一下:

[1] Solaris 的 CC:生成libCalcAdd.so的时候,需要使用-G参数。生成使用so的主程序需要使用-ldl

[2] Mac的g++     : 生成libCalcAdd.so的时候,需要使用-dynamiclib -flat_namespace。生成 使用so的主程序不需要使用另外的选项
[3] Linux的g++   : 生成libCalcAdd.so的时候,需要使用-fPIC -shared。生成使用so的主程序需要使用-ldl.

大家可以下载v3.tar.bz2查看源代码和Makefile。需要先进入Core 中 make -f Makefile.(mac|sun|lnx) core

然后进入plugin中 make -f Makefile.(mac|sun|lnx) plugin

然后在主目录的bin下面运行calc就可以了。

我们还可以试一下,在plugin里面加入一个CalcMax的类,找出数组中的最大值。当我们把这个libAddMax.so放到lib之后,我们主程序就可以运行成calc Max了。

 

这样返回类的指针的设计非常的不错,首先,如果你原有的系统,很可能CalcAdd和CalcMul就是类。这样你几乎不需要修改什么代码;其次,如果你的类CalcAdd也许没有实现那个calc接口,没关系,用Adaptor模式适配一下。当然了,也许你会觉得这里的算法类CalcAdd应该是单件啊,不应该每次都去new。这是因为有的系统,现存的CalcAdd里面有很多pirvate的变量和函数,而且这些函数都不是线程安全的。如果你改成单件类,那么使用过程中你的程序也会线程不安全。至少我们现在的系统就是这样的。所以我选择每次返回一个对象。

 

使用动态库不错,可以完全做到plug-in系统。而且未来完全可以做到热启动,热加载。但是我们先停一停,在下一节,我将给出一种完全不一样的实现plug-in的机制。

 

  • 大小: 16.1 KB
分享到:
评论
15 楼 pengshan 2009-12-21  
为什么不用配置文件来辅助找到你的插件?很多系统都是这么做的。
通过接口来约定协议,通过配置文件来管理和寻找插件。
14 楼 fykyx521 2009-12-15  
第2次看天书
13 楼 mikeandmore 2009-12-14  
ankyhe 写道
在Unix上我可以使用Corba, ICE这样的技术。可是,如果考虑到整个团队,N个项目要公用一个core,我会慎重使用这些技术。我想对Client开发人员要求越低越好。

问题倒不是这个。

component base的软件的开发流程和普通软件绝对不会一样的。并不是因为松耦合,主要是因为成本考虑。而且,会在需求分析上下更大的功夫,因为component的重用很难保证需求的完成,造成requirement和validation部分很昂贵。
12 楼 ankyhe 2009-12-13  
在Unix上我可以使用Corba, ICE这样的技术。可是,如果考虑到整个团队,N个项目要公用一个core,我会慎重使用这些技术。我想对Client开发人员要求越低越好。
11 楼 kimmking 2009-12-12  
觉得lz要的就是com~!
10 楼 mikeandmore 2009-12-07  
七猫 写道
mikeandmore 写道
truekbcl 写道
等0x吧,内存标准确立了就好做了。

这个和0x的memory model没关系。。。

跟内存分配还是有点关系的,如果内存分配不统一,在接口传输,返回的时候会有不少的问题。
我最近在研究corba,那东东复杂得简直超过人的大脑了。

c++ 0x的memory model不涉及分配吧。
主要是说barrier和atomic operation的
9 楼 七猫 2009-12-07  
mikeandmore 写道
truekbcl 写道
等0x吧,内存标准确立了就好做了。

这个和0x的memory model没关系。。。

跟内存分配还是有点关系的,如果内存分配不统一,在接口传输,返回的时候会有不少的问题。
我最近在研究corba,那东东复杂得简直超过人的大脑了。
8 楼 mikeandmore 2009-12-05  
truekbcl 写道
等0x吧,内存标准确立了就好做了。

这个和0x的memory model没关系。。。
7 楼 truekbcl 2009-12-03  
等0x吧,内存标准确立了就好做了。
6 楼 ankyhe 2009-12-03  
mikeandmore 写道
这个。。。我个人觉得组建化和插件化还是有很大差别的。。。

虽然都是松耦合的努力。。。。


这个我完全同意。组件化程度更高,付出的成本也更高。
5 楼 mikeandmore 2009-12-03  
这个。。。我个人觉得组建化和插件化还是有很大差别的。。。

虽然都是松耦合的努力。。。。
4 楼 Arath 2009-12-02  
如果使用动态加载的确不会存在主体rebuild的问题,但是一担放入一个开放环境中,这种事情是无法强行约束的.
3 楼 Arath 2009-12-02  
ankyhe 写道
我们有考虑过这些东西,COM, Corba这种真正的组件技术。但是关键是,如果使用这些组建技术,我们就把系统的难度推倒了一个新的高度。另外,我不明白为什么plugin rebuild。主体需要rebuild。从上面的程序来看,主体程序完全不需要rebuild啊。能否讲的更具体些。


最轻量级的就是用interface的方法,使用queryinterface去获得需要的实现接口.

rebuild的问题是和dll/so的export table的结构以及静态连接的实现有关,如果你在一个dll/so中增加了一个接口有可能会导致原先的export table序列被打乱,那么原来使用静态连接的主体程序就必须rebuild,否则就错了.
当然这个也是可以用过强制约束接口导出的方法来规避. 问题在于不是所有人都会遵守规则.
2 楼 ankyhe 2009-12-02  
我们有考虑过这些东西,COM, Corba这种真正的组件技术。但是关键是,如果使用这些组建技术,我们就把系统的难度推倒了一个新的高度。另外,我不明白为什么plugin rebuild。主体需要rebuild。从上面的程序来看,主体程序完全不需要rebuild啊。能否讲的更具体些。
1 楼 Arath 2009-12-02  
使用so或dll的方式实现插件的松耦和连接可行,但是不是好方法的.
最简单的说如果别人不是动态加载,那么一旦你rebuild后很有可能导致主体也要rebuild.
在实际的系统中,松耦和的实现不在于技术本身,而在于规范的约束,但是反过来,为了减少因为违规使用而带来的问题也通过技术来加以强化是必要的.

做的较好还是windows平台的COM模型,但是即使是COM也会因为开发者的认知问题而导致很多不可预期的问题.

相关推荐

    IDA_plug-in_writing_in_C_or_C++.rar

    IDA_plug-in_writing_in_C_or_C++, 中文和英文两个。IDA_plug-in_writing_in_C_or_C++

    GNU ARM Eclipse Plug-in

    《GNU ARM Eclipse Plug-in:为Eclipse开发环境增添ARM支持》 GNU ARM Eclipse Plug-in是专为Eclipse集成开发环境(IDE)设计的一款扩展插件,由开发者ilg-ul和justxi共同贡献。该插件的主要目的是为了在Eclipse...

    Designing Software Synthesizer Plug-Ins in C++

    Bridging the gap from theory to programming, Designing Software Synthesizer Plug-Ins in C++ For RackAFX, VST3 and Audio Units contains complete code for designing and implementing software ...

    Visual C++ 2015-2019 Redistributable.zip

    开发者使用Visual C++编写的应用程序,通常会依赖于一些特定的库文件,这些库文件在用户计算机上可能并未预装。因此,Visual C++ Redistributable就应运而生,它的主要功能就是提供这些必要的运行时库,使得用户无需...

    Microsoft Visual C++ 2015-2019 运行库合集,包含32位64位

    2. "Microsoft Visual C++ 2015-2019 Redistributable - 14.26.28720.3 x86.exe" 是32位版本的安装程序,同样提供版本号为14.26.28720.3的运行库,适用于32位Windows系统或在64位系统上运行32位程序。 安装这两个...

    Microsoft Visual C++ 2015-2022 Redistributable (x64)

    2. **多版本合并**:2015-2022表明这个包集成了多个Visual C++版本的运行时库,这样用户只需安装一个包,就能满足多个使用不同版本VC++编译的程序的需求。 3. **兼容性**:由于是x64版本,因此它能与64位的Windows...

    S60_Platform_ECom_Plug-in_Examples_v2_0_en

    《S60 Platform ECom Plug-in Examples v2.0_en:Symbian手机程序开发实践指南》 在移动操作系统领域,Symbian以其强大的功能和广泛的设备支持,曾一度成为智能手机开发的重要平台。"S60 Platform ECom Plug-in ...

    gcc-c++-4.4.7-16.el6.x86_64.rpm

    warning: gcc-c++-4.4.7-3.el6.x86_64.rpm: Header V3 RSA/SHA1 Signature, key ID c105b9de: NOKEY error: Failed dependencies: gcc = 4.4.7-3.el6 is needed by gcc-c++-4.4.7-3.el6.x86_64 libstdc++ = 4.4.7-...

    gcc-c++-3.4.6-3.1.x86_64.rpm

    gcc-c++-3.4.6-3.1.x86_64.rpm

    基于libmodbus开源库 C++ modbus-rtu通信测试程序源码

    基于libmodbus开源库 C++ modbus-rtu通信测试程序源码,vs2013平台。此文件为主站程序可实现与从站(从站可以使用Modbus Slave 仿真软件)通信,实现寄存器的读写功能。

    添加Microsoft Visual Studio 解决方案平台PPAPI、NACL、PNACL等的插件

    NACL,全称Native Client,是Google推出的一个技术,允许开发者编写C或C++代码,然后在浏览器沙箱环境中运行,以实现高性能的Web应用。NACL的目标是让Web应用具备接近原生应用的速度和功能。在Visual Studio中添加...

    gcc-c++-3.4.6-8.x86_64.rpm

    Linux安装Oracle先决条件gcc-c++-3.4.6-8.x86_64.rpm

    C++ Templates - The Complete Guide英文版.rar

    《C++ Templates - The Complete Guide》是一本深入探讨C++模板技术的专业书籍,它为开发者提供了全面、详尽的模板知识。模板是C++语言中一个强大的特性,它允许我们编写泛型代码,提高代码的复用性和灵活性。这本书...

    Plug-in GUI-开源

    【标题】:“Plug-in GUI-开源” 开源软件的精髓在于开放源代码,允许开发者自由查看、使用、修改和分发代码。"Plug-in GUI"项目就是这样一个体现,它旨在为程序员构建一个库,以实现他们应用程序的可换肤界面功能...

    Object-Oriented Programming in C++源程序

    Object-Oriented Programming in C++Object-Oriented Programming in C++Object-Oriented Programming in C++

    基于C++MFC-RS485串口通信demo-完整版-代码

    2. **配置串口参数**:接着,使用`Setup`方法设定串口参数,如波特率、数据位、停止位和校验位。例如,设置波特率为9600,数据位为8,停止位为1,无校验。 ```cpp m_serialPort.Setup(9600, 8, CSerialPort::...

    compat-gcc-c++-7.3-2.96.128.i386.rpm

    compat-gcc-c++-7.3-2.96.128.i386.rpm

    gcc-c++-4.4.7-4.el6.i686.rpm等

    gcc-c++-4.4.7-4.el6.i686.rpm cpp-4.4.7-4.el6.i686.rpm gcc-4.4.7-4.el6.i686 .rpm libstdc++-devel-4.4.7-4.el6.i686.rpm libstdc++-4.4.7-4.el6.i686.rpm

    某某学院课程-C++ 基础与深度解析

    某某学院课程-C++ 基础与深度解析

Global site tag (gtag.js) - Google Analytics