第11章 自定义库
当Qt Designer自带的组件无法满足特定需求时,开发者可以通过继承QWidget或其它界面组件类来创建自定义的界面组件。这样的自定义组件可以具有特殊的显示效果和功能。文中以一个示例项目samp11_1为例,详细解释了如何从QWidget类继承创建一个自定义界面组件类TBattery,并展示了如何在程序中使用这个组件。这个组件具有显示电池图形和根据电量更新显示的功能,通过重写paintEven
第十一章 自定义库
当Qt Designer提供的界面组件无法满足实际设计需求时,提供了两种方法来自定义和扩展界面组件:提升法和设计自定义Widget插件。
-
提升法
这种方法主要用于将已有的组件(如QGraphicsView)提升为自定义的类(如TGraphicsView)。这样做可以在原有组件的基础上增加更多的自定义属性和功能,同时保留原有组件的大部分特性和行为。这种方法操作相对简单,适用于对现有组件进行小幅度定制的情况。
-
设计自定义Widget插件
如果需要更深入的自定义或者需要经常使用某个自定义组件,那么可以通过设计自己的Widget插件并安装到Qt Designer的组件面板中。这样,该自定义组件的使用就像Qt自带的界面设计组件一样方便。这种方法更为灵活和强大,但需要更多的开发和配置工作。
自定义插件和库
当Qt Designer自带的组件无法满足特定需求时,开发者可以通过继承QWidget或其它界面组件类来创建自定义的界面组件。这样的自定义组件可以具有特殊的显示效果和功能。
文中以一个示例项目samp11_1为例,详细解释了如何从QWidget类继承创建一个自定义界面组件类TBattery,并展示了如何在程序中使用这个组件。这个组件具有显示电池图形和根据电量更新显示的功能,通过重写paintEvent()函数来实现特殊的界面显示效果。此外,还使用了滑动条来动态设置电池电量,电池图形的显示也会随之更新。
做一个插件
Qt Designer Widget 插件
如何做插件
使用插件
Qt插件API概述:
- Qt提供了两套插件API:高级API用于扩展Qt本身的功能(如数据库驱动、图像格式等),低级API用于扩展自行编写应用程序的功能,特别是创建Qt Designer Widget插件。
插件安装目录与示例:
- 介绍了Qt自带插件的安装目录,并举例说明了如
imageformats和sqldrivers等插件的存放位置及其作用。
创建Qt Designer Widget插件:
- 以创建TPBattery类为例,详细描述了使用Qt Creator创建Qt Designer Widget插件的过程,包括设置插件项目名称、选择开发套件、设置自定义Widget类名称、设置插件名称和资源文件名称等步骤。
编译与安装注意事项:
- 强调了编译插件时编译器的重要性,必须与编译Qt Creator的编译器一致,否则插件无法在Qt Designer中正常显示。
- 说明了Debug和Release模式编译的插件分别适用于不同模式编译的应用程序,并指出Debug模式下编译的插件文件名后会增加“d”字母。
编译结果:
- 描述了编译后生成的文件(如
tpbatteryplugin.dll和tpbatteryplugin.lib),并解释了文件名后缀“d”的含义。
静态库
静态库的使用
当第三方代码以源程序文件的形式提供时,可以直接将文件加入项目进行编译。静态库则是将编译好的目标文件进行归档处理得到的库文件,其特点是编译时会嵌入到可执行文件中,无需在运行时额外加载。
使用静态库时,需要确保静态库的头文件和库文件都已正确添加到项目中,并在编译链接时指定静态库的路径。
大型软件项目中的使用
对于大型软件项目,使用动态库更为合适,因为可以独立更新和维护库文件,而不需要重新编译整个应用程序。例如,VTK和OpenCV等大型软件库通常会被编译为动态库供其他项目使用。
Qt Creator中的操作
Qt Creator支持创建和管理静态库和动态库项目。无论是MSVC还是MinGW等编译器,都可以在Qt Creator中创建相应的库文件。
创建静态库
在Qt中创建静态库的过程与创建共享库类似,但有一些不同之处。下面是创建静态库的步骤:
步骤1:创建一个新项目
- 打开 Qt Creator。
- 选择“新建项目”。
- 选择“库”下的“C++静态库”。
- 点击“继续”,根据提示设置项目名称和路径。
步骤2:设置项目文件(.pro)
在项目的 .pro 文件中,添加以下配置以指定创建静态库:
QT += core
QT -= gui # 如果不需要GUI支持
CONFIG += c++11 # 选择C++11标准
CONFIG += staticlib # 指定为静态库
TARGET = mystaticlibrary # 静态库名称
步骤3:编写代码
创建一个头文件和实现文件。例如,创建 mymath.h 和 mymath.cpp。
mymath.h:
#ifndef MYMATH_H
#define MYMATH_H
class MyMath {
public:
static double add(double a, double b);
};
#endif // MYMATH_H
mymath.cpp:
#include "mymath.h"
double MyMath::add(double a, double b) {
return a + b;
}
步骤4:构建静态库
- 在 Qt Creator 中,选择构建(Build)按钮。
- 选择“构建项目”以生成静态库。
构建完成后,你会在项目的输出目录(通常是 release 或 debug 文件夹)中找到生成的静态库文件(.a 或 .lib)。
步骤5:使用静态库
要在其他 Qt 项目中使用这个静态库,请按照以下步骤:
- 将生成的静态库文件复制到新的项目目录的某个位置。
- 在新项目的
.pro文件中添加如下配置:
INCLUDEPATH += path/to/mylibrary # 头文件路径
LIBS += -Lpath/to/mylibrary -lmystaticlibrary # 库文件路径
在代码中包含头文件并调用库中的函数:
#include "mymath.h"
int main() {
double result = MyMath::add(1.0, 2.0);
// ...
}
使用静态库
步骤1:创建一个新项目
- 打开Qt Creator。
- 选择“新建项目”。
- 选择“库”下的“C++静态库”。
- 点击“继续”,并设置项目名称和路径。
步骤2:设置项目文件(.pro)
在项目的 .pro 文件中,添加以下配置以指定创建静态库:
QT += core
QT -= gui # 如果不需要GUI库
CONFIG += c++11 # 选择C++11标准
CONFIG += static # 指定为静态库
TARGET = mylibrary # 静态库名称
步骤3:编写代码
创建头文件和实现文件。例如,创建 mymath.h 和 mymath.cpp。
mymath.h:
#ifndef MYMATH_H
#define MYMATH_H
class MyMath {
public:
static double add(double a, double b);
};
#endif // MYMATH_H
mymath.cpp:
#include "mymath.h"
double MyMath::add(double a, double b) {
return a + b;
}
步骤4:构建静态库
- 在Qt Creator中,选择构建(Build)按钮。
- 选择“构建项目”以生成静态库。
构建完成后,你会在项目的输出目录(通常是 release 或 debug 文件夹)中找到生成的静态库文件(例如,.a 或 .lib)。
步骤5:使用静态库
要在其他Qt项目中使用这个静态库,请按照以下步骤:
- 将生成的静态库文件复制到新的项目目录的某个位置。
- 在新项目的
.pro文件中添加如下配置:
INCLUDEPATH += path/to/mylibrary # 头文件路径
LIBS += -Lpath/to/mylibrary -lmylibrary # 链接库文件
在代码中包含头文件并调用库中的函数:
#include "mymath.h"
int main(int argc, char *argv[]) {
// 调用静态库中的函数
double result = MyMath::add(1.0, 2.0);
// ...
}
动态库
在Qt中,动态库被称为共享库,在 Windows 平台上就是动态链接库。共享库项目编译后生成后缀为“.dll”的动态链接库文件,.dll文件是在应用程序运行时才被加载和调用的,不像静态库那样在编译期间就被嵌入可执行文件。若更新了.dll 文件版本,只要接口未变,应用程序就可以正常调用动态链接库。
动态库的使用
动态库则是在运行时被链接和使用的库文件,不会嵌入到可执行文件中。这种库的优势在于更新时无需重新编译整个项目,灵活性更高。Qt项目在编译和运行时通常会使用到动态库。
动态库的文件名后缀因平台和编译器不同而有所差异,如Windows平台上的“.dll”文件。在Qt Creator中,可以创建和管理动态库项目。
创建共享库
在Qt中创建共享库(动态链接库)比较简单。下面是创建共享库的步骤:
步骤1:创建一个新项目
- 打开Qt Creator。
- 选择“新建项目”。
- 选择“库”下的“C++共享库”。
- 点击“继续”,并根据提示设置项目名称和路径。
步骤2:设置项目文件(.pro)
在项目的 .pro 文件中,添加以下配置以指定创建动态库:
QT += core
QT -= gui # 如果不需要GUI库
CONFIG += c++11 # 选择C++11标准
CONFIG += shared # 指定为共享库
TARGET = mylibrary # 共享库名称
步骤3:编写代码
创建一个头文件和实现文件。例如,创建 mymath.h 和 mymath.cpp。
mymath.h:
#ifndef MYMATH_H
#define MYMATH_H
#include <QtCore/qglobal.h>
#if defined(MYMATH_LIBRARY)
# define MYMATH_EXPORT Q_DECL_EXPORT
#else
# define MYMATH_EXPORT Q_DECL_IMPORT
#endif
class MYMATH_EXPORT MyMath {
public:
static double add(double a, double b);
};
#endif // MYMATH_H
mymath.cpp:
#include "mymath.h"
double MyMath::add(double a, double b) {
return a + b;
}
步骤4:构建共享库
- 在Qt Creator中,选择构建(Build)按钮。
- 选择“构建项目”以生成共享库。
构建完成后,你会在项目的输出目录(通常是 release 或 debug 文件夹)中找到生成的共享库文件(.dll、.so或.dylib)。
步骤5:使用共享库
要在其他Qt项目中使用这个共享库,请按照以下步骤:
- 将生成的共享库文件复制到新的项目目录的某个位置,或将其添加到系统路径。
- 在新项目的
.pro文件中添加如下配置:
INCLUDEPATH += path/to/mylibrary # 头文件路径
LIBS += -Lpath/to/mylibrary -lmylibrary # 库文件路径
- 在代码中包含头文件并调用库中的函数:
#include "mymath.h"
int main() {
double result = MyMath::add(1.0, 2.0);
// ...
}
使用共享库
调用共享库(也称为动态链接库,DLL或so文件)的方式有两种,分别是显式链接和隐式链接。
1、共享库的调用方式
Qt支持使用共享库来组织和重用代码。共享库可以被多个程序共享,减少内存占用和磁盘空间使用。
显式链接
在显式链接中,程序在运行时加载共享库,并且可以动态地使用库中的函数。你需要使用 dlopen 和 dlsym 来加载和调用库。
**示例:**假设有一个共享库 libmymath.so,其包含一个加法函数。
// mymath.h
extern "C" double add(double a, double b);
// mymath.cpp
#include "mymath.h"
extern "C" double add(double a, double b) {
return a + b;
}
使用显式链接:
#include <QCoreApplication>
#include <QDebug>
#include <dlfcn.h> // 用于Linux共享库
int main(int argc, char *argv[]) {
QCoreApplication a(argc, argv);
void *handle = dlopen("libmymath.so", RTLD_LAZY);
if (!handle) {
qDebug() << "Cannot open library:" << dlerror();
return 1;
}
typedef double (*add_t)(double, double);
add_t add = (add_t)dlsym(handle, "add");
if (!add) {
qDebug() << "Cannot load symbol 'add':" << dlerror();
dlclose(handle);
return 1;
}
double result = add(1.0, 2.0);
qDebug() << "Result:" << result;
dlclose(handle);
return a.exec();
}
2、隐式链接调用共享库
隐式链接是一种在编译时链接共享库的方法。在编译时,编译器会知道库的存在,并根据选择的编译选项来链接库。
示例:设置项目文件(.pro)来使用共享库。
# myproject.pro
TEMPLATE = app
TARGET = MyApp
QT += core
SOURCES += main.cpp
LIBS += -L$$PWD/ -lmymath # 指定库路径和链接库名
然后在代码中直接调用库中的方法:
#include <QCoreApplication>
#include <QDebug>
#include "mymath.h" // 包含头文件
int main(int argc, char *argv[]) {
QCoreApplication a(argc, argv);
double result = add(1.0, 2.0); // 直接调用
qDebug() << "Result:" << result;
return a.exec();
}
总结
- 显式链接适合于需要在运行时决定是否加载库的场景。使用动态加载提供了更大的灵活性。
- 隐式链接在编译时进行,与标准的编译和链接过程相似,通常用于更简单的情况。
更多推荐


所有评论(0)