有两种与插件有关的 API 。一种用来扩展 Qt 本身的功能,如 自定义数据库驱动,图像格式,文本编解码,自定义分格,等等,称为 Higher-Level API 。另一种用于应用程序的功能扩展,称为 Lower-Level API 。前一种是建立在后一种的基础之上的。这里讨论的是后一种,即用来扩展应用程序的 Lower-level API 。
让应用程序支持插件扩展的步骤:
1. 定义一个接口集 ( 只有纯虚函数的类 ) ,用来与插件交流。
2. 用宏 Q_DECLARE_INTERFACE () 将该接口告诉 Qt 元对象系统 。
引自 http://yanboo.ycool.com/post.2749491.html
Qt 的插件机制
yanboo 发表于 2007-11-08 16:42:24
Qt
1 Q_DECLARE_INTERFACE(BrushInterface,"com.trolltech.PlugAndPaint.BrushInterface/1.0")
3. 应用程序中用 QPluginLoader 来装载插件。
4. 用宏 qobject_cast () 来确定一个插件是否实现了接口。
1 QObject *obj = new QTimer;
2 QTimer *timer = qobject_cast<QTimer *>(obj);
写一个插件的步骤:
1. 声明插件类,该类从 QObject 和该插件希望实现的接口继承而来。
2. 用宏 Q_INTERFACES () 将该接口告诉 Qt 元对象系统。
1 class BasicToolsPlugin : public QObject,
2 public BrushInterface,
3 public ShapeInterface,
4 public FilterInterface
5 {
6 Q_OBJECT
7 Q_INTERFACES(BrushInterface ShapeInterface FilterInterface)
8 public:
9 ...
10 };
3. 用宏 Q_EXPORT_PLUGIN2 () 导出插件。
1 Q_EXPORT_PLUGIN2 ( PluginName, ClassName )
4. 用适当的 .pro 文件构建插件。
下面的代码声明了一个接口类:
1 class FilterInterface
2 {
3 public :
4 virtual ~FilterInterface() {}
5 virtual QStringList filters() const = 0 ;
6 virtual QImage filterImage( const QString &filter, const QImage &image, QWidget* parent)= 0;
7 };
8
9 Q_DECLARE_INTERFACE(FilterInterface, "com.trolltech.PlugAndPaint.FilterInterface/1.0" )
这里是实现该接口的插件类的定义:
1 #include <QObject>
2 #include <QStringList>
3 #include <QImage>
4
5 #include <plugandpaint/interfaces.h>
6
7 class ExtraFiltersPlugin : public QObject, public FilterInterface
8 {
9 Q_OBJECT
10 Q_INTERFACES(FilterInterface)
11
12 public :
13 QStringList filters() const ;
14 QImage filterImage( const QString &filter, const QImage &image,
15 QWidget *parent);
16 };
示例 Plug & Paint 的文档详细解释了这一过程。与 Qt Designer 有关的问题请看 Creating Custom Widgets for Qt Designer 。 Echo Plugin Example 是一个关于如何实现扩展 Qt 应用程序的详细示例。
Loading and Verifying Plugins Dynamically
装载插件时。 Qt 库有一些健全检查来确定插件能否被装载和使用。这就可以同时安装多个版本和 Qt 库配置。
· 与较高主版本和 ( 或 ) 次版本号的 Qt 库链接的插件不能被主版本和 ( 或 ) 次版本号较低的库装载。
原理: 一个使用新版 Qt 库的插件可能用了老版本没有的新特征。 Trolltech 有一个只在次版本号升级时添加新功能和 API 的政策,这就是为什么该测试只看主次版本号,而不看补丁号。
· Qt 库和所有插件用一个联编关键字来联编。 Qt 库中的联编关键字被与插件中的联编关键字对照,如果相符,插件就被装载。如果联编关键字不符, Qt 库就拒绝装载该插件。
原理: 见下文对联编关键字的解释。
编译插件来扩展应用程序时,确保插件和应用程序用同样的配置这一点很重要。 这意味着如果应用程序是 release 模式编译的,那么插件也要是 release 模式。
若将 Qt 配置为 debug 和 release 模式都编译,但只在 release 模式下编译应用程序,就要确保你的插件也是在 release 模式下编译的。缺省的,若 Qt 的 debug 编译可用,插件就只在 debug 模式下编译。要强制插件用 release 模式编译,要在工程中添加:
CONFIG += release
这能确保插件兼容应用程序中所用的库版本。
The Build Key
装载插件时, Qt 核对每一个插件的联编关键字要和自己的匹配,以保证所装载的是兼容的插件;任何不匹配的插件不会被装载。
联编关键字包含一下信息:
· Architecture, operating system and compiler.
原理: 在同一编译器的不同版本并不产生二进制兼容代码的场合,编译器的版本也体现在联编关键字里。
· Qt 库的配置。 这个配置是库中所缺少特性的列表,因为这些功能对应的 API 在该库中不可用。
原理: 两个同一版本的 Qt 库的不同配置不是二进制兼容的。装载插件的 Qt 库使用这个 ( 缺少的 ) 特性列表来判断插件是不是二进制兼容的。
注意: 也存在这种情况,插件可以使用在两个不同配置里可用到的特性。但是,编写插件的开发者需要知道,哪些特性在他们的插件和 Qt 的公用工具类中都在被使用。 Qt 库在装载插件时会需要复杂的特性与依赖性的查询确认。这些需求给开发者添了一个不必要的负担,也增加了装载插件的系统开销。为了减少开发时间, 降低应用的运行时消耗,可以使用对联编关键字的简单字符串比较。
· 可选地,可以在配置脚本命令行 指定一个附加的字符串 Optionally, an extra string may be specified on the configure script command line.
原理: 在发布带有应用程序的 Qt 库的二进制时,这给开发者提供了一个编写插件的办法,这样写出来的插件只能被插件链接的那个库所装载。
为了调试可能需要关闭联编关键字校验功能,这可以通过将你运行应用程序的环 境的环境变量 QT_NO_PLUGIN_CHECK 设置为非零来实现。
Static Plugins
插件能被静态地链接到应用程序。如果你创建了 Qt 的静态版本,这仅仅是用来包含 Qt 的预定义插件的一个选项。
当被作为静态库编译时, Qt 提供下面这些静态插件:
Plugin name |
Type |
Description |
qtaccessiblecompatwidgets |
Accessibility |
Accessibility for Qt 3 support widgets |
qtaccessiblewidgets |
Accessibility |
Accessibility for Qt widgets |
qdecorationdefault |
Decorations (Qtopia) |
Default style |
qdecorationwindows |
Decorations (Qtopia) |
Windows style |
qgif |
Image formats |
GIF |
qjpeg |
Image formats |
JPEG |
qmng |
Image formats |
MNG |
qimsw_multi |
Input methods (Qtopia) |
Input Method Switcher |
qwstslibmousehandler |
Mouse drivers (Qtopia) |
tslib mouse |
qgfxtransformed |
Graphic drivers (Qtopia) |
Transformed screen |
qgfxvnc |
Graphic drivers (Qtopia) |
VNC |
qscreenvfb |
Graphic drivers (Qtopia) |
Virtual frame buffer |
qsqldb2 |
SQL driver |
IBM DB2 |
qsqlibase |
SQL driver |
Borland InterBase |
qsqlite |
SQL driver |
SQLite version 3 |
qsqlite2 |
SQL driver |
SQLite version 2 |
qsqlmysql |
SQL driver |
MySQL |
qsqloci |
SQL driver |
Oracle (OCI) |
qsqlodbc |
SQL driver |
Open Database Connectivity (ODBC) |
qsqlpsql |
SQL driver |
PostgreSQL |
qsqltds |
SQL driver |
Sybase Adaptive Server (TDS) |
qcncodecs |
Text codecs |
Simplified Chinese (People's Republic of China) |
qjpcodecs |
Text codecs |
Japanese |
qkrcodecs |
Text codecs |
Korean |
qtwcodecs |
Text codecs |
Traditional Chinese (Taiwan) |
要静态链接这些插件,你的应用程序中要用到宏 Q_IMPORT_PLUGIN () 并且要用 QTPLUGIN 将需要的插件添加到你的编译中。例如,在 main.cpp 中:
#include <QApplication>
#include <QtPlugin>
Q_IMPORT_PLUGIN(qjpeg)
Q_IMPORT_PLUGIN(qgif)
Q_IMPORT_PLUGIN(qkrcodecs)
int main(int argc, char *argv[])
{
QApplication app(argc, argv);
...
return app.exec();
}
应用程序的 .pro 文件中要用下列条目:
QTPLUGIN += qjpeg /
qgif /
qkrcodecs
也可以创建自己的静态库,步骤如下:
1. 在插件的 .pro 文件中添加 CONFIG += static 。
2. 应用程序中用宏 Q_IMPORT_PLUGIN () 。
3. 应用程序的 .pro 文件中用 LIBS 将静态库链接进来。
参见示例 Plug & Paint 和相关的插件 Basic Tools 来获得详情。
The Plugin Cache
为了加速插件的装载和确认,装载插件时收集的信息被缓存到 QSettings 中。这包括插件是否被成功装载的信息,以使 后面的装载操作不用再尝试装载无效的插件。但是,若一个插件的 last modified 时间戳被修改,插件的缓存条目是无效的并且插件会不管缓存条目中的值而被重新装载,同时缓存条目本身也会被新的值替代。
这也意味着每一次插件或任何依赖资源 ( 如共享库 ) 被更新之后时间戳也必须被更新,因为依赖资源可能影响一个插件装载的结果。
有时,开发插件时,需要从插件缓存中移除条目。因为 Qt 用 QSettings 来管理插件缓存,插件的位置是依赖于平台 的;更多关于每一个平台的信息请参看 the QSettings documentation 。
例如, Windows 中这些条目存储在注册表中,每个插件的路径是以下面两个字串中的一个开始的:
HKEY_CURRENT_USER/Software/Trolltech/OrganizationDefaults/Qt Plugin Cache 4.2.debug
HKEY_CURRENT_USER/Software/Trolltech/OrganizationDefaults/Qt Plugin Cache 4.2.false
Debugging Plugins
有许多问题可能影响到插件在应用程序中的正常运转。 许多与插件和应用程序的创建方法不同有关, 通常发生在不同的创建系统和过程中。
下表描述的是开发者创建插件时遇到的问题的常见原因:
Problem |
Cause |
Solution |
应用程序打开插件时插件装载失败且无提 示。 Qt Designer 的 Help|About Plugins 对话框中显示插件库,但这里没有插件。 |
应用程序和插件是在不同模式下创建的 |
使用相同的创建信息;或者在他们的工程 文件中的 CONFIG 变量中添加 debug_and_release 来用 debug 和 release 两种模式创建插件。 |
用有效的插件替代无效插件时装载失败 |
插件缓存中该插进的条目显示原来的插件 不能被卸载,导致 Qt 忽略了替代 |
确保插件的时间戳被更新 或删除插件缓存中的条目 |
也可以用 QT_DEBUG_PLUGINS 环境变量来从 Qt 中获得尝试去装载的每一个插件的诊断信息。在应用程序的运行环境中把该变量的值设置为非零。