CTK插件框架学习-插件注册调用(03)

CTK插件框架学习-新建插件(02)icon-default.png?t=N7T8https://mp.csdn.net/mp_blog/creation/editor/136923735

一、CTK插件组成

  • 接口类:对外暴露的接口,供其他插件调用
  • 实现类:实现接口内的方法
  • 激活类:负责将插件注册到CTK框架中

二、接口、插件、服务三者关系

1、一对一

一个接口类由一个实现类实现,对应一个插件,注册一个服务

参见

CTK插件框架学习-新建插件(02)icon-default.png?t=N7T8https://mp.csdn.net/mp_blog/creation/editor/136923735

2、多对一(多态)

多个接口类由一个实现类实现,对应一个插件,注册多个服务
 

  • 接口类1
    #pragma once
    #include <QObject>class IService1
    {
    public:virtual ~IService1() {}virtual void printf1() = 0;
    };//此宏将当前这个接口类声明为接口,后面的字符串是这个接口的唯一标识。
    Q_DECLARE_INTERFACE(IService1, "zr.IService1")
    

  • 接口类2
    #pragma once
    #pragma once
    #include <QObject>class IService2
    {
    public:virtual ~IService2() {}virtual void printf2() = 0;
    };//此宏将当前这个接口类声明为接口,后面的字符串是这个接口的唯一标识。
    Q_DECLARE_INTERFACE(IService2, "zr.IService2")
    

  • 实现类
    
    ==============================TestPlugin2.ch==================================
    #pragma once#include "IService1.h"
    #include "IService2.h"class ctkPluginContext;
    class TestPlugin2 : public QObject, public IService1, public IService2
    {Q_OBJECT//当一个类继承这个接口类,表明需要实现这个接口类Q_INTERFACES(IService1)Q_INTERFACES(IService2)public:TestPlugin2(ctkPluginContext* contex);virtual void printf1();virtual void printf2();
    };==============================TestPlugin2.cpp=================================#include "TestPlugin2.h"
    #include <qdebug.h>TestPlugin2::TestPlugin2(ctkPluginContext* contex)
    {
    }void TestPlugin2::printf1()
    {qDebug() << "IService1 printf1";
    }void TestPlugin2::printf2()
    {qDebug() << "IService2 printf2";
    }
    

  • 插件测试
#include "CTKPlugin.h"
#include <QtWidgets/QApplication>#include <iostream>
#include <QStyleFactory>
#include <QDir>
#include <QDirIterator>
#include <QDebug>
#include "ctkPluginFrameworkFactory.h"
#include "ctkPluginFramework.h"
#include "ctkPluginException.h"
#include "ctkPluginContext.h"
#include "ctkPluginFrameworkLauncher.h"
#include "../TestPlugin2/IService1.h"
#include "../TestPlugin2/IService2.h"
/*
* 1、注意:Plugin-SymbolicName要满足这里的前缀是:TARGET/META-INF格式。TARGET的名字最好和工程名一致,不然可能出现device not open错误。
* 2、如果CTK初始化、插件安装启动等是在一个类中,则与CTK相关的变量应定义成类的属性,不能是成员变量,否则获取不到服务
* 3、CTK插件组成:
(1)每个插件有自己的注册器Activator,继承自QObject和ctkPluginActivator的一个类,并实现ctkPluginActivator的start、stop函数
(2)每个插件必须有一个资源文件,名称一般与插件名称一致,前缀必须为TARGET/META-INF,例:插件名称/META-INF
(3)每个插件必须添加一个元数据文件,名字必须为MANIFEST.MF,并添加到资源文件中
* 4、QSharedPointer framework这个对象既可以作为对象也可以作为对象指针,但要作为插件框架使用必须要用指针方法调用
* 5、生成的插件名(TARGET)不要有下划线,因为CTK会默认将插件名中的下划线替换成点号,最后导致找不到插件 
*/
int main(int argc, char *argv[])
{QApplication::setAttribute(Qt::AA_EnableHighDpiScaling);QApplication a(argc, argv);a.setApplicationName("ctktest");//Linux下没有名称报错QString path = QCoreApplication::applicationDirPath();// 启动插件工厂ctkPluginFrameworkFactory* ctkFrameWorkFactory = new ctkPluginFrameworkFactory;QSharedPointer<ctkPluginFramework> framework = ctkFrameWorkFactory->getFramework();try {framework->init();framework->start();}catch (const ctkPluginException& e){std::cout << "framework init fail" << std::endl;}QSharedPointer<ctkPlugin> plugin;QDirIterator iter(path + "/plugins/", { "*.dll" }, QDir::NoFilter, QDirIterator::Subdirectories);while (iter.hasNext()) {//qDebug() << iter.next();QString dllPath = iter.next();QUrl url = QUrl::fromLocalFile(dllPath);try{plugin = framework->getPluginContext()->installPlugin(url);//获取MANIFEST.MF中的数据QHash<QString, QString> headers = plugin->getHeaders();ctkVersion version = ctkVersion::parseVersion(headers.value(ctkPluginConstants::PLUGIN_VERSION));QString name = headers.value(ctkPluginConstants::PLUGIN_NAME);}catch (ctkPluginException e) {std::cout << e.message().toStdString() << e.getType() << std::endl;}}try {plugin->start(ctkPlugin::START_TRANSIENT);//表示立即启用插件,不设置参数的话加载后也不会立即打印输出}catch (ctkPluginException e) {std::cout << e.message().toStdString() << e.getType() << std::endl;}//2、测试插件(多个接口一个实现一个服务一个插件)IService1* service1 = NULL;ctkServiceReference ref2 = framework->getPluginContext()->getServiceReference<IService1>();if (ref2){service1 = framework->getPluginContext()->getService<IService1>(ref2);}if (service1){service1->printf1();}IService2* service2 = NULL;ctkServiceReference ref3 = framework->getPluginContext()->getServiceReference<IService2>();if (ref3){service2 = framework->getPluginContext()->getService<IService2>(ref3);}if (service2){service2->printf2();}//ctkPlugin::State sta = plugin->getState();//ctkPluginFrameworkLauncher::stop();//plugin->stop(); //plugin->uninstall();//sta = plugin->getState();CTKPlugin c;c.show();return a.exec();
}

3、一对多

一个接口类由多个实现类实现,对应多个插件,注册一个服务

/*
* 1、通过ctkPluginConstants::SERVICE_RANKING和ctkPluginConstants::SERVICE_ID来调用不同的插件
* 2、插件不同但是在同一个dll内
* 3、插件获取策略:
*		插件容器中id最小的服务,id为插件注册时的SERVICE_RANKING属性
*		插件容器内id相同的情况,返回pid最小的服务
* 4、插件每次调用其他插件时只会生成一个实例,不会因多次调用产生多个服务实例
*/

  • 接口类
    #pragma once
    #include <QObject>/*
    * 1、通过ctkPluginConstants::SERVICE_RANKING和ctkPluginConstants::SERVICE_ID来调用不同的插件
    * 2、插件不同但是在同一个dll内
    * 3、插件获取策略:
    *		插件容器中id最小的服务,id为插件注册时的SERVICE_RANKING属性
    *		插件容器内id相同的情况,返回pid最小的服务
    * 4、插件每次调用其他插件时只会生成一个实例,不会因多次调用产生多个服务实例
    */
    class IService
    {
    public:virtual ~IService() {}virtual void printf() = 0;
    };//此宏将当前这个接口类声明为接口,后面的字符串是这个接口的唯一标识。
    Q_DECLARE_INTERFACE(IService, "zr.IService")
    

  • 实现类1
    #pragma once#include "IService.h"
    #include <QObject>class ctkPluginContext;
    class ServiceImp1 :public QObject, public IService
    {Q_OBJECT//当一个类继承这个接口类,表明需要实现这个接口类Q_INTERFACES(IService)public:ServiceImp1(ctkPluginContext* contex);void printf();};

  • 实现类2
    #pragma once
    #include "IService.h"
    #include <QObject>class ctkPluginContext;
    class ServiceImp2 :public QObject, public IService
    {Q_OBJECT//当一个类继承这个接口类,表明需要实现这个接口类Q_INTERFACES(IService)public:ServiceImp2(ctkPluginContext* contex);void printf();
    };

  • 服务类
    =========================PluginActivator.h=============================
    #pragma once
    #include <qobject.h>
    #include "ctkPluginActivator.h"
    #include "ctkPluginContext.h"
    #include "ServiceImp1.h"
    #include "ServiceImp2.h"class PluginActivator :public QObject, ctkPluginActivator
    {Q_OBJECTQ_INTERFACES(ctkPluginActivator)//向Qt的插件框架声明,希望将xxx插件放入到框架中。Q_PLUGIN_METADATA(IID "TestPlugin3")//向qt框架申明插件(qt5版本)public:PluginActivator();void start(ctkPluginContext *context);void stop(ctkPluginContext *context);
    private:QScopedPointer<ServiceImp1> m_serviceImp1;//智能指针,自动析构回收QScopedPointer<ServiceImp2> m_serviceImp2;//智能指针,自动析构回收
    };=========================PluginActivator.cpp=============================#include "PluginActivator.h"
    #include <QDebug>
    #include "ctkPluginContext.h"
    #include "ctkPluginFrameworkLauncher.h"PluginActivator::PluginActivator()
    {}
    void PluginActivator::start(ctkPluginContext *context)
    {qDebug() << "my plugin3 start";m_serviceImp1.reset(new ServiceImp1(context));m_serviceImp2.reset(new ServiceImp2(context));ctkDictionary dic1;dic1.insert(ctkPluginConstants::SERVICE_RANKING, 1);dic1.insert("name", "ServiceImp1");context->registerService<IService>(m_serviceImp1.get(), dic1);ctkDictionary dic2;dic2.insert(ctkPluginConstants::SERVICE_RANKING, 2);dic2.insert("name", "ServiceImp2");context->registerService<IService>(m_serviceImp2.get(), dic2);}void PluginActivator::stop(ctkPluginContext *context)
    {qDebug() << "my plugin3 stop";Q_UNUSED(context)// Q_UNUSED,如果一个函数的有些参数没有用到、某些变量只声明不使用,但是又不想编译器、编辑器报警报,其他没有什么实际性作用ctkServiceReference  ref1 = context->getServiceReference<IService>();context->ungetService(ref1);m_serviceImp1.reset(NULL);m_serviceImp2.reset(NULL);ctkPlugin::State sta = context->getPlugin()->getState();}

  • 插件测试
    #include "CTKPlugin.h"
    #include <QtWidgets/QApplication>#include <iostream>
    #include <QStyleFactory>
    #include <QDir>
    #include <QDirIterator>
    #include <QDebug>
    #include "ctkPluginFrameworkFactory.h"
    #include "ctkPluginFramework.h"
    #include "ctkPluginException.h"
    #include "ctkPluginContext.h"
    #include "ctkPluginFrameworkLauncher.h"
    #include "../TestPlugin3/IService.h"
    /*
    * 1、注意:Plugin-SymbolicName要满足这里的前缀是:TARGET/META-INF格式。TARGET的名字最好和工程名一致,不然可能出现device not open错误。
    * 2、如果CTK初始化、插件安装启动等是在一个类中,则与CTK相关的变量应定义成类的属性,不能是成员变量,否则获取不到服务
    * 53、CTK插件组成:
    (1)每个插件有自己的注册器Activator,继承自QObject和ctkPluginActivator的一个类,并实现ctkPluginActivator的start、stop函数
    (2)每个插件必须有一个资源文件,名称一般与插件名称一致,前缀必须为TARGET/META-INF,例:插件名称/META-INF
    (3)每个插件必须添加一个元数据文件,名字必须为MANIFEST.MF,并添加到资源文件中
    * 4、QSharedPointer framework这个对象既可以作为对象也可以作为对象指针,但要作为插件框架使用必须要用指针方法调用
    * 5、生成的插件名(TARGET)不要有下划线,因为CTK会默认将插件名中的下划线替换成点号,最后导致找不到插件 
    */
    int main(int argc, char *argv[])
    {QApplication::setAttribute(Qt::AA_EnableHighDpiScaling);QApplication a(argc, argv);a.setApplicationName("ctktest");//Linux下没有名称报错QString path = QCoreApplication::applicationDirPath();// 启动插件工厂ctkPluginFrameworkFactory* ctkFrameWorkFactory = new ctkPluginFrameworkFactory;QSharedPointer<ctkPluginFramework> framework = ctkFrameWorkFactory->getFramework();try {framework->init();framework->start();}catch (const ctkPluginException& e){std::cout << "framework init fail" << std::endl;}//QString dir = QCoreApplication::applicationDirPath();//dir += "/plugins/TestPlugin.dll";//QUrl url = QUrl::fromLocalFile(dir);QSharedPointer<ctkPlugin> plugin;QDirIterator iter(path + "/plugins/", { "*.dll" }, QDir::NoFilter, QDirIterator::Subdirectories);while (iter.hasNext()) {//qDebug() << iter.next();QString dllPath = iter.next();QUrl url = QUrl::fromLocalFile(dllPath);try{plugin = framework->getPluginContext()->installPlugin(url);//获取MANIFEST.MF中的数据QHash<QString, QString> headers = plugin->getHeaders();ctkVersion version = ctkVersion::parseVersion(headers.value(ctkPluginConstants::PLUGIN_VERSION));QString name = headers.value(ctkPluginConstants::PLUGIN_NAME);}catch (ctkPluginException e) {std::cout << e.message().toStdString() << e.getType() << std::endl;}}try {plugin->start(ctkPlugin::START_TRANSIENT);//表示立即启用插件,不设置参数的话加载后也不会立即打印输出}catch (ctkPluginException e) {std::cout << e.message().toStdString() << e.getType() << std::endl;}//3、测试插件(一个接口多个实现一个服务多个插件<一个dll内>)//3.1获取所有服务QList<ctkServiceReference> ref3List = framework->getPluginContext()->getServiceReferences<IService>();foreach (ctkServiceReference var, ref3List){if (var){qDebug() << "service name:" << var.getProperty("name").toString();qDebug() << "service ranking:" << var.getProperty(ctkPluginConstants::SERVICE_RANKING).toLongLong();qDebug() << "service id:" << var.getProperty(ctkPluginConstants::SERVICE_ID).toLongLong();IService* service3 = qobject_cast<IService*>(framework->getPluginContext()->getService(var));if (service3){service3->printf();}}}//3.2获取某些服务IService* service3 = NULL;ref3List = framework->getPluginContext()->getServiceReferences<IService>("(&(name=ServiceImp1))");foreach(ctkServiceReference var, ref3List){if (var){qDebug() << "service name:" << var.getProperty("name").toString();qDebug() << "service ranking:" << var.getProperty(ctkPluginConstants::SERVICE_RANKING).toLongLong();qDebug() << "service id:" << var.getProperty(ctkPluginConstants::SERVICE_ID).toLongLong();IService* service3 = qobject_cast<IService*>(framework->getPluginContext()->getService(var));if (service3){service3->printf();}}}//3.3获取一个服务,由service ranking 和service id决定ctkServiceReference ref = framework->getPluginContext()->getServiceReference<IService>();if (ref) {IService* service3 = qobject_cast<IService*>(framework->getPluginContext()->getService(ref));if (service3){service3->printf();}}//ctkPlugin::State sta = plugin->getState();//ctkPluginFrameworkLauncher::stop();//plugin->stop(); //plugin->uninstall();//sta = plugin->getState();CTKPlugin c;c.show();return a.exec();
    }
    

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/782756.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

文生视频大模型Sora的复现经验

大家好&#xff0c;我是herosunly。985院校硕士毕业&#xff0c;现担任算法研究员一职&#xff0c;热衷于机器学习算法研究与应用。曾获得阿里云天池比赛第一名&#xff0c;CCF比赛第二名&#xff0c;科大讯飞比赛第三名。拥有多项发明专利。对机器学习和深度学习拥有自己独到的…

BFS专题

1、BFS解决FloodFill算法 1、1图像渲染 733. 图像渲染 - 力扣(LeetCode) class Solution {typedef pair<int,int> PII;int dx[4] = {0,0,1,-1};int dy[4] = {1,-1,0,0}; public:vector<vector<int>> floodFill(vector<vector<int>>& i…

RIP环境下的MGRE 综合实验

实验题目及要求&#xff1a; 1.R5为ISP&#xff0c;只能进行IP地址配置&#xff0c;其所有地址均配为公有IP地址 2.R1和R5间使用PPP的PAP认证&#xff0c;R5为主认证方; R2于R5之间使用PPP的chap认证&#xff0c;R5为主认证方&#xff1b; R3于R5之间使用HDLC封装。 3.R1/…

【C++】为什么能实现函数重载

从C语言一路学到C的途中&#xff0c;C语言C语言相比&#xff0c;多了个函数重载&#xff0c;那么函数重载是如何实现的呢&#xff0c;为什么C语言无法支持&#xff0c;在本篇博客中&#xff0c;将会讲解C为何能实现函数重载。 一.编译过程 C能实现函数重载&#xff0c;而C语言不…

QT 二维坐标系显示坐标点及点与点的连线-通过定时器自动添加随机数据点

QT 二维坐标系显示坐标点及点与点的连线-通过定时器自动添加随机数据点 功能介绍头文件C文件运行过程 功能介绍 上面的代码实现了一个简单的 Qt 应用程序&#xff0c;其功能包括&#xff1a; 创建一个 MainWindow 类&#xff0c;继承自 QMainWindow&#xff0c;作为应用程序的…

2024软件设计师备考讲义——UML(统一建模语言)

UML的概念 用例图的概念 包含 <<include>>扩展<<exted>>泛化 用例图&#xff08;也可称用例建模&#xff09;描述的是外部执行者&#xff08;Actor&#xff09;所理解的系统功能。用例图用于需求分析阶段&#xff0c;它的建立是系统开发者和用户反复…

Pyppeteer中Chromium安装步骤

1、下载压缩文件 在官网下载chrome-win.zip文件 2、终端下载pyppeteer 首先在Pycharm终端运行pip install pyppeteer 3、查找文件默认路径 在运行以下代码&#xff0c;找到可执行文件默认路径 import pyppeteer.chromium_downloader print(默认版本是&#xff1a;{}.forma…

牛角工具箱源码 轻松打造个性化在线工具箱

&#x1f389; Whats this&#xff1f; 这是一款在线工具箱程序&#xff0c;您可以通过安装扩展增强她的功能 通过插件模板的功能&#xff0c;您也可以把她当做网页导航来使用~ 觉得该项目不错的可以给个Star~ &#x1f63a; 演示地址 https://tool.aoaostar.com &#x1f…

TCP网络协议栈和Posix网络部分API总结

文章目录 Posix网络部分API综述TCP协议栈通信过程TCP三次握手和四次挥手&#xff08;看下图&#xff09;三次握手常见问题&#xff1f;为什么是三次握手而不是两次&#xff1f;三次握手和哪些函数有关&#xff1f;TCP的生命周期是从什么时候开始的&#xff1f; 四次挥手通信状态…

HarmonyOS实战开发-如何实现一个自定义抽奖圆形转盘

介绍 本篇Codelab是基于画布组件、显式动画&#xff0c;实现的一个自定义抽奖圆形转盘。包含如下功能&#xff1a; 通过画布组件Canvas&#xff0c;画出抽奖圆形转盘。通过显式动画启动抽奖功能。通过自定义弹窗弹出抽中的奖品。 相关概念 Stack组件&#xff1a;堆叠容器&am…

从0开始搭建基于VUE的前端项目(一) 项目创建和配置

准备与版本 安装nodejs(v20.11.1)安装vue脚手架(@vue/cli 5.0.8) ,参考(https://cli.vuejs.org/zh/)vue版本(2.7.16),vue2的最后一个版本vue.config.js的配置详解(https://cli.vuejs.org/zh/config/)element-ui(2.15.14)(https://element.eleme.io/)vuex(3.6.2) (https://…

K8S命令行可视化实验

以下为K8s命令行可视化工具的实验内容&#xff0c;相比于直接使用命令行&#xff0c;可视化工具可能更直观、更易于操作。 Lens Lens是用于监控和调试的K8S IDE。可以在Windows、Linux以及Mac桌面上完美运行。在 Kubernetes 上&#xff1a; 托管地址&#xff1a;github/lensa…

机器人运动控制

一、基础 1.1 矢量速度和旋转速度 矢量速度用来控制运动方向&#xff0c;任何一个方向都可以看成x、y、z三轴方向的合。单位规定是m/s。 旋转速度用来控制旋转方向&#xff0c;可以看成x、y、z三轴方向旋转的合。单位规定是pi/s。 速度消息包&#xff0c;可以在ROS Index上搜…

助力福建新型职业农民培育 北方天途推进无人机植保应用培训

为加强新型职业农民的职业培育&#xff0c;扩展新型农民的知识范围和专业技术水平&#xff0c;推进农业供给侧结构性改革。日前&#xff0c;在农业部门的大力支持下&#xff0c;北方天途航空和宁德天禾科技服务携手为福建省农民朋友开展了植保无人机驾驶员的应用培训。福建省农…

网页布局案例 浮动

这里主要讲浮动 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>Title</title><style>*{padding: 0;margin: 0;}.header{height: 40px;background-color: #333;}.nav{width: 1226px;heig…

深入理解数据结构(2):顺序表和链表详解

文章主题&#xff1a;顺序表和链表详解&#x1f331;所属专栏&#xff1a;深入理解数据结构&#x1f4d8;作者简介&#xff1a;更新有关深入理解数据结构知识的博主一枚&#xff0c;记录分享自己对数据结构的深入解读。&#x1f604;个人主页&#xff1a;[₽]的个人主页&#x…

机器学习——降维算法-奇异值分解(SVD)

机器学习——降维算法-奇异值分解&#xff08;SVD&#xff09; 在机器学习中&#xff0c;降维是一种常见的数据预处理技术&#xff0c;用于减少数据集中特征的数量&#xff0c;同时保留数据集的主要信息。奇异值分解&#xff08;Singular Value Decomposition&#xff0c;简称…

csp资料

头文件 #include <bits/stdc.h> using namespace std isdigit(c); isalpha(c); switch(type){case value : 操作 } continue;//结束本轮循环 break;//结束所在的整个循环tips: //除法变乘法来算 //减法变加法 num1e42;//"1e4"表示10的4次方//用于移除容器中相…

某国投集团知识竞赛活动方案

一、抽签分组办法 1.抽签&#xff1a;参赛队伍赛前进行抽签分组。 2.分组&#xff1a;全部报名参赛队伍按照抽签顺序分为4组&#xff0c;每组7支队伍进行预赛&#xff0c;9月16日上午1、2组进行初赛&#xff0c;9月16日下午3、4组进行初赛。每组决出的前三名进入决赛。 二、初…

二维码门楼牌管理应用平台建设:引领现代化小区管理新篇章

文章目录 前言一、二维码门楼牌管理应用平台概述二、三维动态单体化技术的优势三、二维码门楼牌管理应用平台的应用场景四、展望未来 前言 随着城市化的快速推进&#xff0c;现代化小区如雨后春笋般涌现&#xff0c;对小区管理的效率和智能化提出了更高要求。二维码门楼牌管理…