`
cloverprince
  • 浏览: 130181 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

插件问题回答第2题 另解

阅读更多
问题贴:http://cloverprince.iteye.com/blog/481307

引用

2. 现有一个主程序用C++语言写成。现在要允许第三方开发人员编写扩展的类,约定第三方开发的类必须包含一个继承自某个已知类(如class FooPlugin)的子类,名称不限。如果要求第三方的类必须与主程序的二进制代码分开发布,把dll或so丢在某个文件夹内即可被动态装载使用,应如何实现?



回答:

除了直接使用操作系统提供的接口以外,还可以使用Qt提供的插件系统。我们需要一个纯虚类,并用Q_DECLARE_INTERFACE宏声明该接口。插件模块中,具体类继承该接口,并用Q_EXPORT_PLUGIN2输出该具体类。主程序中,用QPluginLoader装载该模块,用QPluginLoader::instance()获得实例,用qobject_cast<>()转换成接口类型。


适用范围:

Qt 4.x


实现:

接口定义:hellointerface.h
#ifndef HELLOINTERFACE_H
#define HELLOINTERFACE_H

#include <QtPlugin>

class HelloInterface
{
public:
    virtual ~HelloInterface() {}

    virtual QString getName() const = 0; // 取得名字
    virtual void setName(QString name) = 0; // 设置名字

    virtual void greet() = 0; // 打招呼
};

 Q_DECLARE_INTERFACE(HelloInterface,
                     "com.trolltech.PlugAndPaint.BrushInterface/1.0")

#endif // HELLOINTERFACE_H



这里只做一个插件
helloworldplugin.h
#ifndef HELLOWORLDPLUGIN_H
#define HELLOWORLDPLUGIN_H

#include "HelloWorldPlugin_global.h"
#include "../hellointerface.h"

class HELLOWORLDPLUGINSHARED_EXPORT HelloWorldPlugin :
        public QObject,
        public HelloInterface
{
    Q_OBJECT
    Q_INTERFACES(HelloInterface)

private:
    QString name;

public:
    virtual QString getName() const;
    virtual void setName(QString name);

    virtual void greet();


};

#endif // HELLOWORLDPLUGIN_H


helloworldplugin.c:
#include "helloworldplugin.h"

#include <cstdio>

QString HelloWorldPlugin::getName() const
{
    return this->name;
}

void HelloWorldPlugin::setName(QString name)
{
    this->name = name;
}

void HelloWorldPlugin::greet()
{
    std::printf("Hello, %s\n",this->name.toStdString().c_str());
}

Q_EXPORT_PLUGIN2(helloworldplugin, HelloWorldPlugin)

这里的Q_EXPORT_PLUGIN2宏输出HelloWorldPlugin具体类。第一个参数需要和.pro文件中的TARGET字段相同。


以上是插件。下面是主程序
main.cpp:
#include <QtCore/QCoreApplication>
#include <QPluginLoader>
#include <QDir>

#include "hellointerface.h"

#include <cstdio>

const char* PLUGINS_PATH="plugins";

int main(int argc, char *argv[])
{

    QDir pluginsDir(PLUGINS_PATH);

    foreach(QString fileName, pluginsDir.entryList()) {
        if(fileName.endsWith(".so")) {
            printf("Loading %s\n",fileName.toStdString().c_str());

            QPluginLoader loader(pluginsDir.absoluteFilePath(fileName));
            HelloInterface *plugin =
                    qobject_cast<HelloInterface*>(loader.instance());

            plugin->setName("wks");
            plugin->greet();
        }
    }
    return 0;
}



编译:

插件按照Qt Library的方式编译;主程序按照Qt Application方式编译即可。


运行:

文件结构:

引用
.
|-- PluginTest
`-- plugins
    `-- libhelloworldplugin.so



执行结果:
引用
Loading libhelloworldplugin.so
Hello, wks



总结:
1. main程序并不了解plugins目录中有多少插件。在运行时列举目录。 
2. main程序对每个plugins文件(比如叫libhelloworldplugin.so)的了解只有: 
- libhelloworldplugin.so定义了一个类,实现了那个已知接口,并用Q_EXPORT_PLUGIN2输出。 

分享到:
评论

相关推荐

    2015-16-2《高等数学》(经管类)期末考试试卷1

    - 第2题是优化问题,通过求解拉格朗日乘数问题找到最大利润的广告费用分配策略。 - 第3题涉及到未来值和现值的概念,需要计算一次性投资的现值以匹配未来的养老金支付。 - 第4题要求计算一个闭区域上的二重积分,...

    第10章重积分自测1

    第二题要求计算锥面被柱面截取部分的曲面面积,可能需要结合积分和微分方程来解。第三题是二重积分的直接计算,注意积分区域的定义。第四题是计算曲面、平面和两个边界围成的闭区域的三重积分。第五题是关于旋转曲面...

    android面试题.rar

    "android面试题.rar"这个压缩包很可能包含了关于Android开发的常见面试问题和解答,旨在帮助求职者准备面试。以下是根据标题和描述推测出的一些可能涉及的Android面试知识点,以及相关的详细解释: 1. **Android...

    第十一届蓝桥杯大赛软件赛决赛_CB1

    例如,试题A“美丽的2”,作为结果填空题,其设计初衷在于考查选手对简单算法的理解和应用。在这个问题中,参赛者需要计算从公元1年到2020年中所有包含数字2的年份数量。看似简单,实际上却需要参赛者能够灵活运用...

    PyPI 官网下载 | kolibri_exercise_perseus_plugin-0.9.3.tar.gz

    在Python的世界中,PyPI(Python Package Index)是官方的第三方Python软件包仓库,它为开发者提供了一个集中发布和分发Python模块的平台。PyPI上的资源“kolibri_exercise_perseus_plugin-0.9.3.tar.gz”是一个用于...

    NIIT在线考试SM3-MT1,最新的

    3. **SM3-MT1**:这是一个特定的课程或考试模块,SM3可能是课程系列中的第三阶段,而MT1可能代表该阶段的第一个模块或考试。具体的课程内容可能涵盖软件开发、项目管理、网络技术、数据库管理等IT领域的某一专题。 ...

Global site tag (gtag.js) - Google Analytics