用PyQt进行Python图形界面的程序的开发的入门指引


Posted in Python onApril 14, 2015

一般来说,选择用于应用程序的 GUI 工具箱会是一件棘手的事。使用 Python(许多语言也一样)的程序员可以选择的 GUI 工具箱种类繁多,而每个工具箱都有各自的优缺点。有些速度比其它工具箱快,有些比较小;有些易于安装,有些更适合于跨平台使用(对于这一点,还要指出,有些支持您需要满足的特定特性)。当然,各种库都相应具有各种许可证。

对于 Python 程序员而言,缺省的 GUI 选择是 Tk(通过 Tkinter 绑定)— 其原因显而易见。Tkinter 和闲置的 IDE 是由 Python 创始人编写的,它们是作为大多数 Python 分发版的缺省选择而出现的。标准 Python 文档讨论了 Tkinter,但没有涉及任何其它 GUI 绑定。这是故意的!至少可以这么认为,如果 Tk 和 Tkinter 不是这么糟糕,程序员就没有理由去寻找替代品了。要诱导 Python 程序员放弃缺省选择,那么工具箱必须提供额外的东西。PyQt 就是这样一个工具箱。

PyQt 所具有的优点远远超过了 Tkinter(它也有几个缺点)。Qt 和 PyQt 速度都很快;Qt 和 PyQt 的设计完全是面向对象的;Qt 提供了一个设计良好的窗口构件集合,它比 Tk 所提供的要大得多。就其缺点而言,Qt 的许可证受到的限制比许多工具箱(至少在非 Linux 平台方面)都多;正确安装 Qt 和 PyQt 常常会很复杂;另外,Qt 是一个相当大的库。PyQt 应用程序的用户将需要设法完成安装 Qt 和 PyQt,这使分发变得很困难。(请阅读本文后面的 用于其它语言的 Qt 绑定。)

PyQt 严格遵循 Qt 的发放许可。特别是,它可用于 UNIX/X11 平台上的 GPL,并可用于 Zaurus 上的 Qt Palmtop Environment 环境,还存在用于较老的 Qt 版本的免费(free-as-in-free-beer)Windows 软件包。PyQt 的商业许可证可用于 Windows。

对于本文而言,PyQt 有一个方面优于许多其它工具箱,它值得我们特别关注。Qt 使用一种称为 信号/插槽(signals/slots)的机制在窗口构件(以及其它对象)之间传递事件和消息。这种机制完全不同于包括 Tkinter 在内的大多数工具箱所用的回调(callback)机制。使用信号/插槽以灵活且可维护的方式控制对象间通信要比使用脆弱的回调风格容易得多。应用程序越大,Qt 的这个优势就越重要。

本文的作者之一 Boudewijn Rempt 已经出版了一本有关使用 PyQt 进行应用程序开发的书籍。 GUI Programming with Python: QT Edition(请参阅 参考资料)显示了如何设计和开发完整的 GUI 应用程序,其中包括从最初的构思到分发的全过程。
样本应用程序

要显示信号/插槽和回调之间的反差,我们提供了一个写着玩玩的应用程序,它使用 Tkinter 和 PyQt。尽管实际上 PyQt 版本对于这个基本程序并不更简单,但是它已经演示了 PyQt 应用程序更好的模块性和可维护性。

应用程序包括四个窗口构件:

  1.     “Quit”按钮(用来与整个应用程序通信)
  2.     “Log Timestamp”按钮(用于窗口构件间的消息)
  3.     文本区域,显示可滚动的已记录日志的时间戳记列表
  4.     消息窗口构件,显示已记录日志的时间戳记数

在 Tkinter 中,我们可以这样实现应用程序:

清单 1. Logger.py Tkinter 应用程序

#!/usr/bin/python
import sys, time
from Tkinter import *
class Logger(Frame):
  def __init__(self):
    Frame.__init__(self)
    self.pack(expand=YES, fill=BOTH)
    self.master.title("Timestamp logging application")
    self.tslist = []
    self.tsdisp = Text(height=6, width=25)
    self.count = StringVar()
    self.cntdisp = Message(font=('Sans',24),
                textvariable=self.count)
    self.log = Button(text="Log Timestamp",
             command=self.log_timestamp)
    self.quit = Button(text="Quit", command=sys.exit)
    self.tsdisp.pack(side=LEFT)
    self.cntdisp.pack()
    self.log.pack(side=TOP, expand=YES, fill=BOTH)
    self.quit.pack(side=BOTTOM, fill=BOTH)
  def log_timestamp(self):
    stamp = time.ctime()
    self.tsdisp.insert(END, stamp+"\n")
    self.tsdisp.see(END)
    self.tslist.append(stamp)
    self.count.set("% 3d" % len(self.tslist))
if __name__=='__main__':
  Logger().mainloop()

这个 Tk 版本使用了 log_timestamp() 方法作为按钮的 command= 参数。 这个方法需要依次单独操作它要影响的所有窗口构件。如果我们想更改按钮按下的效果(例如还要记录时间戳记),那么这个风格就很脆弱。通过继承您可以实现这一点:

清单 2. StdOutLogger.py Tkinter 增强

class StdOutLogger(Logger):
  def log_timestamp(self):
    Logger.log_timestamp(self)
    print self.tslist[-1]

但是这个子类的作者需要相当精确地理解 Logger.log_timestamp() 已经做了什么;而且除非通过在子类中完全重写 .log_timestamp() 方法并且不调用父方法,否则没有办法 除去消息。

一个非常基本的 PyQt 应用程序总有一些样本代码,这些代码在哪里都相同,Tkinter 代码也是这样。但是,当我们进一步研究设置应用程序所需的代码,以及显示窗口构件的代码时,区别就显现出来了。

清单 3. logger-qt.py PyQt 应用程序

 

#!/usr/bin/env python
import sys, time
from qt import * # Generally advertised as safe
class Logger(QWidget):
  def __init__(self, *args):
    QWidget.__init__(self, *args)
    self.setCaption("Timestamp logging application")
    self.layout = QGridLayout(self, 3, 2, 5, 10)
    self.tsdisp = QTextEdit(self)
    self.tsdisp.setMinimumSize(250, 300)
    self.tsdisp.setTextFormat(Qt.PlainText)
    self.tscount = QLabel("", self)
    self.tscount.setFont(QFont("Sans", 24))
    self.log = QPushButton("&Log Timestamp", self)
    self.quit = QPushButton("&Quit", self)
    self.layout.addMultiCellWidget(self.tsdisp, 0, 2, 0, 0)
    self.layout.addWidget(self.tscount, 0, 1)
    self.layout.addWidget(self.log, 1, 1)
    self.layout.addWidget(self.quit, 2, 1)
    self.connect(self.log, SIGNAL("clicked()"),
           self.log_timestamp)
    self.connect(self.quit, SIGNAL("clicked()"),
           self.close)
  def log_timestamp(self):
    stamp = time.ctime()
    self.tsdisp.append(stamp)
    self.tscount.setText(str(self.tsdisp.lines()))
if __name__ == "__main__":
  app = QApplication(sys.argv)
  app.connect(app, SIGNAL('lastWindowClosed()'), app,
         SLOT('quit()'))
  logger = Logger()
  logger.show()
  app.setMainWidget(logger)
  app.exec_loop()

通过创建布局管理器, Logger 类开始工作了。布局管理器在任何 GUI 系统中都是一个很复杂的主题,但是 Qt 的实现使之变得简单。在大多数情况下,您会使用 Qt Designer 创建一般的 GUI 设计,随后可将它用于生成 Python 或 C++ 代码。然后您可以使生成的代码生成子类,以添加功能。

但是在这个示例中,我们选择手工创建布局管理器。窗口构件被置于网格的各个单元中,或者可以跨多个单元放置。在 Tkinter 需要命名参数的地方,PyQt 就不允许它们。这是一个很重要的差异,它经常会使在两种环境中工作的人们无所适从。

所有 Qt 窗口构件都可以和 QString 对象很自然地一起工作,而不能和 Python 字符串或 Unicode 对象一起工作。幸运的是,转换是自动的。如果您在 Qt 方法中使用了字符串或 Unicode 参数,那么它将自动转换成 QString。不能进行反向转换:如果您调用了一个返回 QString 的方法,那么您获得的是 QString。

应用程序中最有趣的部分是我们将 clicked 信号连接到功能的位置。一个按钮连接到了 log_timestamp 方法;另一个连接到了 QWidget 类的 close 方法。
图 1. logger-qt 的屏幕快照
logger-qt 的屏幕快照

用PyQt进行Python图形界面的程序的开发的入门指引

现在我们想将日志记录添加到这个应用程序的标准输出。 这十分容易。我们可以使 Logger 类生成子类,或者为了演示,创建简单的独立函数:

清单 4. logger-qt.py PyQt 增强

def logwrite():
  print(time.ctime())
if __name__ == "__main__":
  app = QApplication(sys.argv)
  app.connect(app, SIGNAL('lastWindowClosed()'), app,
        SLOT('quit()'))
  logger = Logger()
  QObject.connect(logger.log, SIGNAL("clicked()"), logwrite)
  logger.show()
  app.setMainWidget(logger)
  app.exec_loop()

从上述代码我们可以看到,这就是将 log QPushButton 的 clicked() 信号连接到新函数的事情。注:信号也可以将任何数据传送到它们所连接的插槽,尽管在这里我们没有显示这样的示例。

如果您不想调用原始方法,那么可以从插槽 disconnect 信号,例如通过在 logger.show() 行之前添加以下行:

清单 5. logger-qt.py PyQt 增强

QObject.disconnect(logger.log, SIGNAL("clicked()"),
          logger.log_timestamp)

现在将不再更新 GUI。

用于 Python 的其它 GUI 绑定

PyQt 在给定实例中可能不是很有用,可能是许可证状态问题,也可能是平台可用性问题(或者,可能因为再分发很困难,例如大小很大)。由于这个原因(也为了比较),我们想指出一些用于 Python 的其它流行 GUI 工具箱。

    Anygui
    Anygui 实际上不是 GUI 工具箱,而是一个作用于大量工具箱(甚至是令人惊奇的象 curses 和 Java/Jython Swing 那样的工具箱)的抽象包装器。在编程风格方面,使用 Anygui 类似于使用 Tkinter,但是要选中这个底层工具箱,要么自动进行,要么进行配置调用。Anygui 很好用,因为它允许应用程序未经更改就可以运行在差异很大的平台上(但因此它支持受支持工具箱的“最低级公共特性”)。
    PyGTK
    PyGTK 绑定包装了 GPL 下使用的 GTK 工具箱,它是流行的 Gnome 环境的基础。GTK 在根本上是 X Window 工具箱,但是它还有 Win32 的 beta 级支持和 BeOS 的 alpha 级支持。在常规范例中,PyGTK 对窗口构件使用回调。绑定存在于 GTK 和 大量编程语言之间,而不仅仅是 Qt,或甚至是 Tk。
    FXPy
    Python 绑定 FXPy 包装了 FOX 工具箱。FOX 工具箱已经被移植到大多数类 UNIX 平台上,以及 Win32 上。 与大多数工具箱类似,FOX 和 FXPy 都使用回调范例。FOX 由 LGPL 特许。
    wxPython
    这个绑定包装了 wxWindows 工具箱。 与 FOX 或 GTK 类似,wxWindows 被移植到 Win32 和类 UNIX 平台上(但是没有移植到 MacOS、OS/2、BeOS 或其它“次要”平台上 — 尽管它对 MacOSX 的支持是 alpha 级的)。在范例方面,wxPython 接近回调风格。wxPython 对继承结构的关注程度高于大多数其它工具箱,而且它使用“事件”,而不是回调。但是本质上,事件仍旧被连接到单个方法上,随后可能需要作用于各种窗口构件。
    win32ui
    win32ui 属于 win32all 软件包,它包装了 MFC 类。很显然,这个工具箱是特定于 Win32 的库。MFC 实际上不只是 GUI 工具箱,它还使用各种范例的混合。对于想创建 Windows 应用程序的读者而言,与其它工具箱相比,win32ui 会让您“更接近于实质”。

从其它语言使用 Qt

如同 Python,从大量其它编程语言使用 Qt 工具箱是可能的。如果可以自由选择,我们会首选 Python,而不是其它语言。诸如公司政策以及与其它代码库连接之类的外部约束可以决定编程语言的选择。Qt 的原始语言是 C++,但也有用于 C、Java、Perl 和 Ruby 的绑定。就与 Python 示例的比较而言,让我们讨论一下用 Ruby 和 Java 写着玩玩的应用程序。

Ruby/Qt 在用法上十分类似于 PyQt。这两种语言具有相似的动态性和简明性,所以除了拼写上的差别外,其代码很类似:

清单 6. HelloWorld.rb Qt2 应用程序

#!/usr/local/bin/ruby
require 'qt2'
include Qt2
a = QApplication.new([$0] + ARGV)
hello = QPushButton.new('Hello world!')
hello.resize(100, 30)
a.connect( hello, QSIGNAL('clicked()'), a, QSLOT('quit()'))
a.setMainWidget(hello)
hello.show
a.exec

Java 总是比脚本编制语言要冗长一点,但是基本部分都相同。一个同等功能的最小 qtjava 应用程序类似于:

清单 7. HelloWorld.java Qt2 应用程序

import org.kde.qt.*;
public class HelloWorld {
 public static void main(String[] args)
 {
  QApplication myapp = new QApplication(args);
  QPushButton hello = new QPushButton("Hello World", null);
  hello.resize(100,30);
  myapp.connect(hello, SIGNAL("clicked"),
         this, SLOT("quit()"));
  myapp.setMainWidget(hello);
  hello.show();
  myapp.exec();
  return;
 }
 static {
  System.loadLibrary("qtjava");
  try {
    Class c = Class.forName("org.kde.qt.qtjava");
  } catch (Exception e) {
    System.out.println("Can't load qtjava class");
  }
 }
}

PyQt 是一个吸引人和快速的接口,它将 Qt 工具箱和 Python 编程语言集成在一起。除了该工具箱提供的种类繁多的窗口构件外,Qt 所用的信号/插槽编程风格在生产能力和可维护性方面都要优于大多数其它 GUI 工具箱所用的回调风格。

Python 相关文章推荐
在Django的模型和公用函数中使用惰性翻译对象
Jul 27 Python
深入理解 Python 中的多线程 新手必看
Nov 20 Python
Python 2与Python 3版本和编码的对比
Feb 14 Python
Python第三方Window模块文件的几种安装方法
Nov 22 Python
python 在某.py文件中调用其他.py内的函数的方法
Jun 25 Python
python基于paramiko将文件上传到服务器代码实现
Jul 08 Python
python虚拟环境的安装和配置(virtualenv,virtualenvwrapper)
Aug 09 Python
python中的Elasticsearch操作汇总
Oct 30 Python
pytorch使用tensorboardX进行loss可视化实例
Feb 24 Python
实现ECharts双Y轴左右刻度线一致的例子
May 16 Python
用python实现名片管理系统
Jun 18 Python
手把手教你用Django执行原生SQL的方法
Feb 18 Python
使用C语言来扩展Python程序和Zope服务器的教程
Apr 14 #Python
用Python中的wxPython实现最基本的浏览器功能
Apr 14 #Python
Python中SOAP项目的介绍及其在web开发中的应用
Apr 14 #Python
Python中的XML库4Suite Server的介绍
Apr 14 #Python
Python pickle模块用法实例
Apr 14 #Python
使用Python的PEAK来适配协议的教程
Apr 14 #Python
Python全局变量操作详解
Apr 14 #Python
You might like
第四章 php数学运算
2011/12/30 PHP
PHP 快速排序算法详解
2014/11/10 PHP
php多进程并发编程防止出现僵尸进程的方法分析
2020/02/28 PHP
jquery1.4后 jqDrag 拖动 不可用
2010/02/06 Javascript
解决jQuery插件tipswindown与hintbox冲突
2010/11/05 Javascript
关于jquery input textare 事件绑定及用法学习
2013/04/03 Javascript
JavaScript实现找出字符串中第一个不重复的字符
2014/09/03 Javascript
JavaScript中数组的合并以及排序实现示例
2015/10/24 Javascript
详解js中call与apply关键字的作用
2016/11/21 Javascript
简单实现JS上传图片预览功能
2017/04/14 Javascript
移动端web滚动分页的实现方法
2017/05/05 Javascript
微信小程序中页面FOR循环和嵌套循环
2017/06/21 Javascript
JS实现的简单下拉框联动功能示例
2018/05/11 Javascript
JavaScript ES6 Class类实现原理详解
2020/05/08 Javascript
vue 接口请求地址前缀本地开发和线上开发设置方式
2020/08/13 Javascript
Python中的pass语句使用方法讲解
2015/05/14 Python
Python写入CSV文件的方法
2015/07/08 Python
最近Python有点火? 给你7个学习它的理由!
2017/06/26 Python
浅谈Python处理PDF的方法
2017/11/10 Python
Python3爬虫教程之利用Python实现发送天气预报邮件
2018/12/16 Python
python for和else语句趣谈
2019/07/02 Python
Ubuntu+python将nii图像保存成png格式
2019/07/18 Python
Python容器使用的5个技巧和2个误区总结
2019/09/26 Python
H5新属性audio音频和video视频的控制详解(推荐)
2016/12/09 HTML / CSS
End Clothing美国站:英国男士潮牌商城
2018/04/20 全球购物
opencv实现图像平移效果
2021/03/24 Python
旺仔牛奶广告词
2014/03/20 职场文书
导游个人求职信
2014/04/25 职场文书
创业融资计划书
2014/04/25 职场文书
中国梦演讲稿开场白
2014/08/28 职场文书
普通党员对照检查材料
2014/09/24 职场文书
勤俭节约主题班会
2015/08/13 职场文书
深度好文:50条没人告诉你的人生经验,句句精辟
2019/08/22 职场文书
flex弹性布局详解
2022/03/20 HTML / CSS
vue3引入highlight.js进行代码高亮的方法实例
2022/04/08 Vue.js
Win11怎么解除儿童账号限制?Win11解除微软儿童账号限制方法
2022/07/07 数码科技