10分钟带你了解轻量级插件框架x3py

2023-11-20

写在前面

  • 由于本人目前主要从事的是Windows客户端开发方面的工作,所以本文介绍x3py的侧重点也是从客户端程序开发者方面叙述的。本文主要参考整理自x3py的官方Wiki,修正了一些官方示例中的错误。有兴趣的同学可以直接阅读原文。

设计目的

  • x3py 作为一个轻量级的C++插件框架,面向C++开发人员,首要目标是能快速容易的开发出中小型的软件、软件以插件形式模块化设计。其插件既可灵活组合到各个系统,又能单独拆开使用和测试。

  • 另外,x3py 可在Win/Linux/MacOSX等平台上编译运行,可使用VS或GCC编译,具备基本的跨平台兼容性。框架力求最简化,核心设计目标为“实用、简洁”。

客户端开发引入此框架目的

  • 我们客户端方面引入x3py主要是因为他即可以实现类似COM组件开发效果,又具有跨平台兼容性。这样我们可以很方便的实现一个跨平台的组件化客户端开发框架。在引入x3py后,我们实现的核心框架层次从下到上可以如下所示。

    • FrameWork:最底层的基础框架层,提供最基本的公共函数,如日志、网络处理、基础ui框架等。x3py框架属于这一层。
    • MiddlePlugin:中间插件层,主要封装了提供给应用层使用的公共插件,提供公共的接口给应用层调用。在引入x3py使得客户端面向接口开发后,我们程序员的主要工作就在开发这些公共插件。
    • Application:应用层,应用层框架就比较简单了,主要负责插件的加载和接口调用,一些简单的ui处理等。

插件原理

  • 使用x3py所实现的C++插件,即可以是动态链接库文件(DLL/SO),也可以是可执行程序文件;通过在插件文件中实现统一的导出函数,使得其他模块可以自动通过这些导出函数访问其接口。

一、插件工作原理

插件原理图

  • 插件系统主要有两种使用形式,一是在C++的主程序或动态库中使用C++插件,另一种是在Python等其他编程语言中使用C++插件。
插件管理器(x3manager)
  • 首先,x3manager也是一个插件,按照统一标准实现了插件的导出函数。x3manager插件的主要作用是在多个插件之间进行中介联系,从而允许插件之间能够互相访问而不直接发生联系,各个插件动态库加载后自动向x3manager登记。例如要创建一个对象,通过插件管理器来查询是在哪个插件中实现了该类,从而创建出对象、访问接口功能。

  • 在x3py插件框架中,x3manager不负责插件动态库的加载管理,所以在C++的主程序或动态库中使用C++插件时,需要先加载各个插件动态库,然后就可使用插件接口。由于框架提供了多种加载方法,这样更灵活、职责更清晰。

二、插件调用过程

  • 在插件实现时,通过包含 pluginimpl.h 文件,自动实现了 x3InternalCreate、x3InitPlugin、x3FreePlugin、x3CreateObject 这四个导出函数,通过导出函数就可在插件之间建立联系。下面将说明这四个导出函数之间的联系。
1、加载插件过程

加载插件的序列图

  • 使用 x3LoadLibrary 函数加载动态库文件。使用此函数而不是WinAPI的 LoadLibraryA 函数,是为了能实现跨平台同时确保自动调用 x3InitPlugin 函数进行插件初始化。

  • x3InitPlugin 函数用于在插件加载时进行初始化、向管理器注册(x3RegisterPlugin)此插件的所有类ID(CLSID串)。通过在插件CPP文件中包含 pluginimpl.h 文件,自动实现了 x3InitPlugin。插件动态库被多次加载时不会多次重复初始化,这是在 x3InitPlugin 函数中通过计数来保证的。

  • x3InitializePlugin 函数由各个插件自行实现,用来完成附加的初始化操作,由 x3InitPlugin 函数调用。

2、卸载插件过程

卸载插件的序列图

  • 使用 x3FreeLibrary 函数卸载动态库文件。使用此函数而不是WinAPI的 FreeLibrary 函数,同样也是是为了能实现跨平台同时确保自动调用 x3FreePlugin 函数进行插件注销。

  • x3FreePlugin 函数用于在插件卸载时进行对象释放、向管理器注销(x3UnregisterPlugin)此插件的信息。通过在插件CPP文件中包含 pluginimpl.h 文件,自动实现了 x3FreePlugin。

  • x3UninitializePlugin 函数也是由各个插件自行实现,用来完成附加的资源释放操作,由 x3FreePlugin 函数调用。

3、创建插件接口指针过程

创建插件接口指针的序列图

  • 通过使用智能指针模板类 x3::Object 来创建插件接口指针,自动处理引用计数。

  • x3::Object模板类通过x3::createObject函数去调用 x3manager中的x3CreateObject 来创建对象。如果插件自身的 x3InternalCreate 不能创建对象,则再转给x3manager,由后者根据类ID查找对应插件,然后调用对应插件的 x3InternalCreate 函数。

如何使用插件接口

  • 使用接口比较简单,在C++代码中包含相应的接口头文件,然后使用智能指针模板类 x3::Object 来创建对象或者从一个对象得到其他接口的对象。

  • 下面的例子说明了如何创建对象、从一个对象得到其他接口的对象、使用接口函数:

#include <plsimple/isimple.h>               // 包含接口文件
#include <plsimple/isimple2.h>              // 包含其他接口文件

int test()
{
    x3::Object<ISimple> obj(clsidSimple);     // 给定类ID创建对象

    if (obj)                  // 检查是否创建成功
    {
        obj->add(1, 2);       // 调用接口函数,就像普通指针一样
    }

    x3::Object<ISimple2> other(obj.p());  // 从已有对象得到其他接口的实例吗,p()是获取obj的接口类指针

    // other = obj.p();            // 除了拷贝构造,也可以使用赋值形式

    if (other.valid())  
    {
        std::vector<int> nums{1,2,3,4,5};
        other->add(nums);   // 调用新接口的函数
    }

    // 会自动释放对象
}

  • 对于函数返回值是接口对象或函数形参是接口对象的情况,为了避免在头文件中包含所用接口的头文件,可以使用 x3::AnyObject ,需要具体某个接口的对象时再转换。下面的例子就是插件的接口文件中 createSimple() 返回的是 x3::AnyObject,这样在此接口头文件中就可以不包含ISimple1的头文件:
#ifndef X3_EXAMPLE_ISIMPLE3_H
#define X3_EXAMPLE_ISIMPLE3_H

#include <objptr.h>

const char* const clsidSimple3 = "94071767-ba6b-4769-9eb4-2ebf469218f3";

class ISimple3 : public x3::IObject
{
    X3DEFINE_IID(ISimple3);

    // 形参用AnyObject可避免包含其他接口定义文件,可在注释中说明实际接口名
    virtual void useSimple(const x3::AnyObject& obj) = 0;

    // 返回值用AnyObject可避免包含其他接口定义文件,可在注释中说明实际接口名
    virtual x3::AnyObject createSimple() = 0;

    // 而不是 virtual x3::Object<ISimple1> createSimple() = 0;
};

#endif

  • 此函数的实现示例如下,其中包含了如何使用AnyObject得到特定接口对象的例子:
void CSimple::useSimple(const x3::AnyObject& obj)
{
    x3::Object<ISimple> myobj(obj);
    ASSERT(myobj);

    x3::Object<ISimple> sample(createSimple());
    ASSERT(sample);
}

x3::AnyObject CSimple::createSimple()
{
    return x3::Object<ISimple>(clsidSimple);
    // x3::Object<ISimple> simp(clsidSimple);
    // do something with simp
    // return simp;
}

如何加载插件

1、同时使用多个插件的简单加载方法

#include <portability/x3port.h>     // 相当于#include<windows.h>
#include <nonplugin/useplugins.h>   // 包含辅助加载类,一个工程内只能包含一次

int main()
{
    // 多个插件文件名,以NULL结尾
    const char* plugins[] = {
        "x3manager.pln", "plsimple.pln", "observerex.pln", NULL
    };
    // 自动加载和卸载插件,插件在程序文件的plugins子目录下
    x3::AutoLoadPlugins autoload(plugins, "plugins");

    // 可以使用接口了
    return test();
}

2、仅使用一个插件的简单加载方法

#include <portability/x3port.h>     // 相当于#include<windows.h>

// 插件在程序文件的plugins子目录下,如果不定义则插件与程序同目录
// 注意末尾有目录分隔符,在包含useplugin.h前定义,如果不定义则为默认的空串
#define PLUGIN_PATH  "plugins/"

// 要使用的插件文件名,没有目录和后缀名(.pln)
#define PLUGIN_NAME  "plsimple"

// 包含辅助加载类,一个工程内只能包含一次
#include <nonplugin/useplugin.h>

// 可以使用接口了
int main()
{
    return test();
}

3、遍历目录下的所有插件

#include <nonplugin/scanplugins.h>

int main()
{
    // 默认遍历 plugins 目录,加载其中的插件文件
    x3::loadPlugins();

    // 可以使用接口了
    int ret = test();

    // 卸载所有插件,后加载的先卸载
    x3::unloadPlugins();

    return ret;
}

4、原生加载方式

#include <portability/portimpl.h>

HMODULE modules[10] = { NULL };

int main()
{
    const char* plugins[] = { 
        "plugins/x3manager.pln", 
        "plugins/plsimple.pln", 
        "plugins/observerex.pln",
        NULL
    };
    int count = 0;

    for (int i = 0; plugins[i]; i++)
    {
        modules[count] = x3LoadLibrary(plugins[i]);
        if (modules[count])
            count++;
    }
    
    // 可以使用接口了
    int ret = test();

    while (--count >= 0)
    {
        x3FreeLibrary(modules[count]);
    }

    return ret;
}

namespace x3 {
class IObject;
bool createObject(const char* clsid, long iid, IObject** p)
{
    typedef bool (*F)(const char*, long, IObject**);
    F f = (F)GetProcAddress(modules[0], "x3CreateObject");
    return f && f(clsid, iid, p);
}
} // x3

5、静态链接库方式

这种方式使用的比较少,就不具体介绍了,官方有一个具体的Wiki介绍,有兴趣的同学可以自己看看看。

#include <portability/x3port.h>
#include <module/pluginimpl.h>
#define XUSE_LIB_PLUGIN
#include <module/modulemacro.h>
XDEFINE_EMPTY_MODULE()

#ifdef _MSC_VER
#pragma comment(lib, "libpln1.lib")
#endif

extern const x3::ClassEntry* const classes_libpln1;

const x3::ClassEntry* const x3::ClassEntry::classes[] = {
        s_classes, classes_libpln1, NULL
    };

// 可以使用接口了
int main()
{
    return test();
}
  • C++插件后缀名为 .pln ,而不是 .dll.so.plugin.dll ,其原因是在不同操作系统中用统一的后缀名,避免条件定义;同时避免与普通动态库混淆,这样遍历目录批量加载插件时可跳过普通动态库文件。

  • 在VS中,我们可以通过工程属性–链接器–输出文件来设置编译出的插件dll后缀名为 .pln
    VS中后缀名设置

如何实现插件

一、定义接口

  • 首先定义接口类,此接口类从 x3::IObject 派生,包含纯虚函数,使用 X3DEFINE_IID(接口名) 定义接口ID。X3DEFINE_IID 和 x3::IObject 定义在 iobject.h  中,但通常我们包含 objptr.h 更方便些,下面是一个接口头文件isimple.h:
#ifndef X3_EXAMPLE_ISIMPLE_H
#define X3_EXAMPLE_ISIMPLE_H

#include <objptr.h>

class ISimple : public x3::IObject
{
    X3DEFINE_IID(ISimple);
    virtual int add(int a, int b) const = 0;
    virtual int subtract(int a, int b) const = 0;
};

#endif

二、定义类UID

  • 一个插件类如果要让外部模块能创建对象实例,需要指定插件类的全局唯一标识信息,一般采用GUID串来标识插件类。通常一个插件类UID常量可以定义成如下形式:
const char* const clsidSimple = "94071767-ba6b-4769-9eb4-2ebf469289f3";

  • 类UID既可以定义在单独的头文件中,也可以就定义在接口的头文件中。我们的习惯做法是:如果一个插件类定义了多个接口文件,则将类UID定义在单独的头文件中,之后每个接口文件包含此头文件;如果一个插件类仅定义了一个接口,为方便起见则将类UID就定义在接口的头文件中。

注意:类UID不可重复,否则将会覆盖已向管理器注册的类,导致其他模块无法创建原来类的接口对象。

三、实现接口

  • 接下来,我们来实现isimple.h中定义的接口,使用 X3BEGIN_CLASS_DECLARE 等宏来指定一个类实现了哪些接口。这里的CSimple是一个实现类的例子。其中构造函数和析构函数申明为保护函数,所有实现的接口函数申明为私有函数,这是推荐方式,但不是必须的。这样做是为了明确表明不允许直接实例化和delete销毁,也不允许直接调用实现类的接口函数。
#ifndef X3_EXAMPLE_SIMPLE_IMPL_H
#define X3_EXAMPLE_SIMPLE_IMPL_H

#include <plsimple/isimple.h>   // 包含接口定义

class CSimple : public ISimple  // 从接口派生
{
    X3BEGIN_CLASS_DECLARE(CSimple, clsidSimple) // 指定类ID
        X3DEFINE_INTERFACE_ENTRY(ISimple)       // 指定实现的接口
    X3END_CLASS_DECLARE()
protected:
    CSimple();
    virtual ~CSimple();

private:
    virtual int add(int a, int b) const;
    virtual int subtract(int a, int b) const;
};

#endif

  • 一个实现类可以实现一个或多个接口,还可以从已有的实现类派生以便继承所有的接口实现(实现继承),或者重载已实现的接口函数。下面是实现继承的一个例子:
#ifndef X3_EXAMPLE_SIMPLE_IMPL2_H
#define X3_EXAMPLE_SIMPLE_IMPL2_H

#include "plsimple.h"           // 包含基实现类
#include <plsimple/isimple2.h>  // 包含附加接口
#include <plsimple/isimple3.h>

class CSimple2
    : public CSimple            // 从基实现类派生
    , public ISimple2           // 从附加接口派生
    , public ISimple3
{
    X3BEGIN_CLASS_DECLARE(CSimple2, clsidSimple)
        X3DEFINE_INTERFACE_ENTRY(ISimple2) // 指定实现的接口
        X3DEFINE_INTERFACE_ENTRY(ISimple3)
        X3USE_INTERFACE_ENTRY(CSimple)     // 继承已实现的所有接口
    X3END_CLASS_DECLARE()
protected:
    CSimple2() {}

private:
    virtual int add(const std::vector<int>& nums) const;
    virtual x3::AnyObject createSimple();
};

#endif

四、注册实现类

  • 因为插件中的实现类一般不直接用于实例化对象,而是通过 x3::Object 智能指针模板类来创建对象的。所以,我们需要在插件中注册可供实例化的类以及说明此类是否是单例。

  • 注册实现类的通常做法是在工程的一个.cpp文件(通常为 module.cpp)中,包含 pluginimpl.h 和 modulemacro.h 文件,然后使用 XBEGIN_DEFINE_MODULE 等宏来申明注册哪些实现类。下面是示例:

#include <module/plugininc.h>
#include <module/pluginimpl.h>      // 实现插件的导出函数
#include <module/modulemacro.h>     // 注册实现类的宏定义

#include "plsimple.h"               // 包含实现类

XBEGIN_DEFINE_MODULE()
    XDEFINE_CLASSMAP_ENTRY(CSimple) // 注册普通实现类或单实例类
    XDEFINE_CLASSMAP_ENTRY_Singleton(YourSingletonClass)
XEND_DEFINE_MODULE_DLL()            // 插件动态库

OUTAPI bool x3InitializePlugin()    // 插件加载时执行,用于额外初始化
{
    return true;
}

OUTAPI void x3UninitializePlugin()  // 插件卸载时执行,用于释放额外数据
{
}

  • 其中 x3InitializePlugin() 和 x3UninitializePlugin() 函数由自己实现,用于额外的初始化和释放操作。这两个函数可以在module.cpp或其他.cpp文件中实现,但是在同一个插件中只能实现一次。

  • 如果某个插件不实现任何接口,只使用其他插件的接口,则可以在module.cpp中使用下面更简单的形式:

#include <module/pluginimpl.h>
#include <module/modulemacro.h>
XDEFINE_EMPTY_MODULE()
  • 由于在VS的.cpp文件中一般都会包含stdafx.h,因此 module/plugininc.h 文件无论包含与否都可。使用VS时可以不需要 module/plugininc.h 和 interface\core\portability 目录。

事件驱动机制(Observer)

x3py官方也提供了事件驱动机制,其的主要用途是在不同模块之间实现松耦合(相互隔离)、将复杂的调用流程分离为多个独立操作片断、实现功能扩展点。但其实现比较简单,只支持最多两个参数。所以,我们可以引入信号槽或者其他方法来替换。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

10分钟带你了解轻量级插件框架x3py 的相关文章

随机推荐

  • 单目标追踪——【Transformer】MixFormer: End-to-End Tracking with Iterative Mixed Attention

    目录 文章侧重点 网络结构 MAM Mixed Attention Module MixFormer 论文 代码 文章侧重点 本文的出发点是认为现有的多阶段Siamese追踪框架 特征提取 特征融合 边界框预测 的前两步 特征提取 特征融合
  • 搬运工~看到一个很有意思的python程序

    coding utf 8 import 二炮 Class 核武 二炮 二炮任务 默认小日本 def init self self 核武状态 二炮 NB status def status self return self 核武状态 def
  • Python编程中的for循环语句学习教程

    本文来源于公众号 csdn2299 喜欢可以关注公众号 程序员学府 这篇文章主要介绍了Python编程中的for循环语句学习教程 是Python入门学习中的基础知识 需要的朋友可以参考下 Python for循环可以遍历任何序列的项目 如一
  • 12306 图形验证码闲谈

    验证码是一个非常有意思的问题 它的目的是区分输入者是人还是机器 这个问题本质上是一个图灵测试 推荐电影 模仿游戏 验证码即是一种简单高效的验证方法 由CMU的教授于2000年左右创造 后来此牛人又将零星的验证码收集起来 转化为巨大的生产力
  • 养生产品如何进行线上推广?产品线上推广的渠道有哪些?

    随着时间的推移 中国人口老龄化越来越多 并且由于生活水平不断提高 居民收入富裕 越来越多的人对于自身健康问题越来越关注 健康养身行业将会迎来蓬勃发展 养生行业内的企业公司现在可以提前进行市场布局 抓住时机发展壮大 那么 健康养生产品如何进行
  • Virtual Box安装时出现严重错误的解决方法

    之前安装了Vitual Box 卸载后尝试了网上很多方法 查了很多资料 最后成功解决 记录一下方法 如下情况 1 我们先下载Mirosoft Visual C 2019 输入如下网址 选择第一个内容 这里我们向下划 找到如下图 下载对应的版
  • 关于conda使用环境未被激活问题

    若在安装Anaconda之后 出现 Warning This Python interpreter is in a conda environment but the environment has not been activated 那
  • 【Liunx】进程控制和父子进程

    文章目录 1 进程和程序 1 1进程和程序的概念 1 2单道和多道程序设计 1 3进程状态的转换 1 3 1进程的状态切换 1 3 2MMU 内存管理单元的作用 1 3 4PCB 进程控制块 的认识 1 3 5获取环境变量 2 控制进程 2
  • go语言基础-----03-----流程控制、函数、值传递、引用传递、defer函数

    1 流程控制 这里只讲 for range 语句 这个关键字 主要用于遍历 用来遍历数组 slice map chan 例如 package main import fmt func main str hello world 中国 for
  • 【阶段二】Python数据分析Pandas工具使用05篇:数据预处理:数据的规范化

    本篇的思维导图 数据预处理 数据的规范化 数据标准化 归一化 处理是数据挖掘的一项基础工作 不同评价指标往往具有不同的量纲 数值间的差别可能很大 不进行处理可能会影响数据分析的结果 为了消除指标之间的量纲和取值范围差异的影响 需要进行标准化
  • C语言的不完整类型和前置声明

    声明与定义 Declaration and Definition 开始这篇文章之前 我们先弄懂变量的declaration和definition的区别 即变量的声明和定义的区别 一般情况下 我们这样简单的分辨声明与定义的区别 建立存储空间的
  • Black Duck----安装 Synopsys Detect,运行你的第一次

    Synopsys Detect 也称为 Detect 可以在您的桌面图形用户界面 GUI 或命令行 CLI 上运行 虽然 Detect 运行有许多变化 但运行将执行以下三个步骤 首先 Detect 将使用项目的包管理器来派生该包管理器已知的
  • Anaconda中安装并运行tensorflow

    Anaconda中安装并运行tensorflow 创建环境 选择自己喜欢的名字和需要的python环境进行创建 这里我创建了一个名字为tensorflow的虚拟环境 蓝色框住的地方为这个虚拟环境的路径 记住 有用的 现在这样就是在创建中了
  • 【CV with Pytorch】第 7 章 :图像异常检测

    机器学习的研究使我们进入了研究各种模式和行为的过程 它使我们能够构建可以研究封闭环境的模型 预测能力通常遵循模型训练过程 这是我们在训练模型时需要经常问的一个重要问题 还有另一个问题需要回答 多少数据足以帮助模型理解分布 以便我们有一个好的
  • 恢复误删除的数据库数据-Oracle

    Oracle数据误删快速恢复 文章目录 1 创建测试数据 2 数据未提交 rollback回滚 3 数据已提交 3 1 通过scn恢复数据 3 2 通过时间恢复数据 1 创建测试数据 第一步创建测试表 create table cctest
  • 区域生长法快速入门,不带种子点的选取

    这里实现的区域生长法 是最原始的区域生长法 基本原理是判断像素点的强度值是和种子点的强度值差是否小于阈值 如果小于阈值则被标记 实现过程中利用了栈的先进后出的思想 将8邻域中符合生长要求的点压入栈 然后依次取出 然后在取出的点的基础上对8邻
  • QT信号和槽以结构体为参数传递复杂数据

    QT 的信号和槽机制能十分方便的用来传输数据 但是如果数据种类比较多 分类比较多的时候 就需要更好地更高效的来传递数据的方法 以结构体作为参数是个很不错的选择 这几天写的程序正好需要以结构体来作为参数 但是网上搜的资料很少 讲的也不详细 我
  • chatgpt赋能python:用Python对图片进行分类

    用Python对图片进行分类 在如今的数字时代 图片分类是一个越来越常见的任务 特别是在搜索引擎优化中 图片分类可以让搜索引擎更容易地找到特定类型的图片 并在相关的搜索中以更高的排名显示它们 在本文中 我们将介绍如何使用Python来分类图
  • 理解java反射机制

    Java反射机制 java反射机制是各种框架的基础 因为有了反射 可以实现类的动态加载 只有类在运行期间用到时才加载到内存中 实现动态编程 对于反射机制的理解 主要需要理解类加载的过程 这是非常重要的一点 通过java视频教程整理文档如下
  • 10分钟带你了解轻量级插件框架x3py

    写在前面 由于本人目前主要从事的是Windows客户端开发方面的工作 所以本文介绍x3py的侧重点也是从客户端程序开发者方面叙述的 本文主要参考整理自x3py的官方Wiki 修正了一些官方示例中的错误 有兴趣的同学可以直接阅读原文 设计目的