2019-02-18 信号和信号槽

PyQt5:PyQt5 信号与槽(PyQt5的事件处理机制)

一、事件

  在事件模型,有三个参与者:事件源、事件目标、事件对象。

  事件源:状态发生改变的对象,它产生事件 Source_Obj

  事件目标:是想要被通知的对象 Target_Obj

  事件对象:封装了事件源中的状态变化 Evnet_Obj

  PyQt5有一个独一无二的信号和槽机制来处理事件。信号和槽用于对象之间的通信。当指定事件发生,一个事件信号会被发射。槽可以被任何Python脚本调用。当和槽连接的信号被发射时,槽会被调用。调用示意图如图1所示:

图1

 二、信号和槽(或槽函数)

在Qt中,每一个QObject对象和PyQt中所有继承自QWidget的控件(这些都是QObject的子对象)都支持信号与槽机制。当信号发射时,连接的槽函数将会自动执行。在PyQt 5中信号与槽通过object.signal.connect()方法连接。

PyQt的窗口控件类中有很多内置信号,开发者也可以添加自定义信号。信号与槽具有如下特点。

一个信号可以连接多个槽。

一个信号可以连接另一个信号。

信号参数可以是任何Python类型。

一个槽可以监听多个信号。

信号与槽的连接方式可以是同步连接,也可以是异步连接。

信号与槽的连接可能会跨线程。

信号可能会断开。


在GUI编程中,当改变一个控件的状态时(如单击了按钮),通常需要通知另一个控件,也就是实现了对象之间的通信。在早期的GUI编程中使用的是回调机制,在Qt中则使用一种新机制——信号与槽。在编写一个类时,要先定义该类的信号与槽,在类中信号与槽进行连接,实现对象之间的数据传输。信号与槽机制示意图如图1所示。


图2

当事件或者状态发生改变时,就会发出信号。同时,信号会触发所有与这个事件(信号)相关的函数(槽)。信号与槽可以是多对多的关系。一个信号可以连接多个槽,一个槽也可以监听多个信号。

关于PyQt API中信号与槽的更详细解释,可以参考官方网站: http://pyqt.sourceforge.net/Docs/PyQt5/signals_slots.html?highlight=pyqtsignal#PyQt5.QtCore.pyqtSignal

三、高级自定义信号与槽

所谓高级自定义信号与槽,指的是我们可以以自己喜欢的方式定义信号与槽函数,并传递参数。自定义信号的一般流程如下:

  (1)定义信号。

  (2)定义槽函数。

  (3)连接信号与槽函数。

  (4)发射信号。

(1)定义信号

 通过类成员变量定义信号对象。使用 pyqtSignal()方法

class MyWidget(QWidget):

    # 无参数的信号

    Signal_NoParameters = pyqtSignal()   

    # 带一个参数(整数)的信号     

    Signal_OneParameter = pyqtSignal(int)       

    # 带一个参数(整数或者字符串)的重载版本的信号       

    Signal_OneParameter_Overload = pyqtSignal([int],[str]) 

    # 带两个参数(整数,字符串)的信号     

        Signal_TwoParameters = pyqtSignal(int,str)   

    # 带两个参数([整数,整数]或者[整数,字符串])的重载版本的信号     

    Signal_TwoParameters_Overload = pyqtSignal([int,int],[int,str])


import sys

import PyQt5.QtWidgets as PQW

import PyQt5.QtCore as PQC

class MyWidget(PQW.QWidget):

# 无参数的信号

Signal_NoParameters = PQC.pyqtSignal()

# 带一个参数(整数)的信号

Signal_OneParameter = PQC.pyqtSignal(int)

# 带一个参数(整数或者字符串)的重载版本的信号

Signal_OneParameter_Overload = PQC.pyqtSignal([int],[str])

# 带两个参数(整数,字符串)的信号

Signal_TwoParameters = PQC.pyqtSignal(int,str)

# 带两个参数([整数,整数]或者[整数,字符串])的重载版本的信号

Signal_TwoParameters_Overload = PQC.pyqtSignal([int,int],[int,str])

(2)定义槽函数

  定义一个槽函数,它有多个不同的输入参数。槽函数就是普通类中的函数或方法。

class MyWidget(PQW.QWidget):  #接上例程序,同一个类MyWidget。

    def setValue_NoParameters(self): 

        '''无参数的槽函数'''passdef setValue_OneParameter(self,nIndex): 

        '''带一个参数(整数)的槽函数'''passdef setValue_OneParameter_String(self,szIndex): 

        '''带一个参数(字符串)的槽函数'''passdef setValue_TwoParameters(self,x,y): 

        '''带两个参数(整数,整数)的槽函数'''passdef setValue_TwoParameters_String(self,x,szY): 

        '''带两个参数(整数,字符串)槽函数'''pass


(3)连接信号与槽函数

  通过connect方法连接信号与槽函数或者可调用对象。

app = QApplication(sys.argv) 

widget = MyWidget()  # 连接无参数的信号widget.Signal_NoParameters.connect(self.setValue_NoParameters )                                          # 连接带一个整数参数的信号widget.Signal_OneParameter.connect(self.setValue_OneParameter)                                        # 连接带一个整数参数,经过重载的信号widget.Signal_OneParameter_Overload[int].

    connect(self.setValue_OneParameter)                              # 连接带一个整数参数,经过重载的信号widget.Signal_OneParameter_Overload[str].

    connect(self.setValue_OneParameter_String )                    # 连接一个信号,它有两个整数参数widget.Signal_TwoParameters.connect(self.setValue_TwoParameters )                                        # 连接带两个参数(整数,整数)的重载版本的信号widget.Signal_TwoParameters_Overload[int,int].

    connect(self.setValue_TwoParameters )                      # 连接带两个参数(整数,字符串)的重载版本的信号widget.Signal_TwoParameters_Overload[int,str].

    connect(self.setValue_TwoParameters_String )             

widget.show() 


(4)发射信号

  通过emit()方法发射信号。

class MyWidget(QWidget): 

    def mousePressEvent(self, event): 

        # 发射无参数的信号        self.Signal_NoParameters.emit()

        # 发射带一个参数(整数)的信号self.Signal_OneParameter.emit(1)

        # 发射带一个参数(整数)的重载版本的信号self.Signal_OneParameter_Overload.emit(1)

        # 发射带一个参数(字符串)的重载版本的信号self.Signal_OneParameter_Overload.emit("abc")

        # 发射带两个参数(整数,字符串)的信号self.Signal_TwoParameters.emit(1,"abc")

        # 发射带两个参数(整数,整数)的重载版本的信号self.Signal_TwoParameters_Overload.emit(1,2)

        # 发射带两个参数(整数,字符串)的重载版本的信号self.Signal_TwoParameters_Overload.emit (1,"abc")


(5)实例

 View Code


运行结果如下:

signal1 emit

signal2 emit,value: 1signal3 emit,value: 1 text

signal4 emit,value: [1, 2, 3, 4]

signal5 emit,value: {'name':'wangwu','age':'25'}

signal6 emit,value: 1 text

signal6 overload emit,value: text

 四、使用自定义参数

  在PyQt编程过程中,经常会遇到给槽函数传递自定义参数的情况,比如有一个信号与槽函数的连接是:

button1.clicked.connect(show_page)

  我们知道对于clicked信号来说,它是没有参数的;对于show_page函数来说,希望它可以接收参数。希望show_page函数像如下这样:

def show_page(self, name):

    print(name,"  点击啦")

   于是就产生一个问题——信号发出的参数个数为0,槽函数接收的参数个数为1,由于0<1,这样运行起来一定会报错(原因是信号发出的参数个数一定要大于槽函数接收的参数个数)。解决这个问题就是:自定义参数的传递。

  有两种解决方法,其中一种解决方法是使用lambda表达式。其完整代码如下:

fromPyQt5.QtWidgetsimport QMainWindow, QPushButton , QWidget , QMessageBox, QApplication, QHBoxLayoutimport sysclass WinForm(QMainWindow):

    def__init__(self, parent=None):

        super(WinForm, self).__init__(parent)

        button1 = QPushButton('Button 1')

        button2 = QPushButton('Button 2')

        button1.clicked.connect(lambda: self.onButtonClick(1))

        button2.clicked.connect(lambda: self.onButtonClick(2))

        layout = QHBoxLayout()

        layout.addWidget(button1)

        layout.addWidget(button2)

        main_frame = QWidget()

        main_frame.setLayout(layout)

        self.setCentralWidget(main_frame)

    def onButtonClick(self, n):

        print('Button {0} 被按下了'.format(n))

        QMessageBox.information(self, "信息提示框",'Button {0} clicked'.format(n))if__name__=="__main__":

    app = QApplication(sys.argv)

    form = WinForm()

    form.setGeometry(300,300,600,400)

    form.show()

    sys.exit(app.exec_())

 运行效果如下:

图3

  这里重点解释onButtonClick()函数是怎样处理从两个按钮传来的信号的。使用lambda表达式传递按钮数字给槽函数,当然也可以传递其他任何东西,甚至是按钮控件本身(假设槽函数打算把传递信号的按钮修改为不可用的话)。

  另一种解决方法是使用functools中的partial函数。实例代码如下:

from PyQt5.QtWidgets import QMainWindow, QPushButton , QWidget , QMessageBox, QApplication, QHBoxLayout

import sys

from functools import partial

class WinForm(QMainWindow):

def __init__(self, parent=None):

super(WinForm, self).__init__(parent)

button1 = QPushButton('Button 1')

button2 = QPushButton('Button 2')

# button1.clicked.connect(lambda: self.onButtonClick(1))

# button2.clicked.connect(lambda: self.onButtonClick(2))

button1.clicked.connect(partial(self.onButtonClick, 1))

button2.clicked.connect(partial(self.onButtonClick, 2))

layout = QHBoxLayout()

layout.addWidget(button1)

layout.addWidget(button2)

main_frame = QWidget()

main_frame.setLayout(layout)

self.setCentralWidget(main_frame)

def onButtonClick(self, n):

print('Button {0} 被按下了'.format(n))

QMessageBox.information(self, "信息提示框", 'Button {0} clicked'.format(n))

if __name__ == "__main__":

app = QApplication(sys.argv)

form = WinForm()

form.setGeometry(300,300,600,400)

form.show()

sys.exit(app.exec_())

   运行效果和上图一样。采用哪种方法好一点呢?这属于风格问题,笔者比较喜欢使用lambda表达式,因为其条理清晰,而且灵活。

五、装饰器信号与槽

  所谓装饰器信号与槽,就是通过装饰器的方法来定义信号和槽函数。具体的使用方法如下:

@PyQt5.QtCore.pyqtSlot(参数)def on_发送者对象名称_发射信号名称(self, 参数):

        pass

  这种方法有效的前提是下面的函数已经被执行:

QtCore.QMetaObject.connectSlotsByName(QObject)


   在上面代码中,“发送者对象名称”就是使用setObjectName函数设置的名称,因此自定义槽函数的命名规则也可以看成:on + 使用setObjectName设置的名称 + 信号名称。接下来看具体的使用方法,完整代码如下:

fromPyQt5import QtCorefromPyQt5.QtWidgetsimport QApplication  ,QWidget ,QHBoxLayout , QPushButtonimport sysclass CustWidget(QWidget):

    def__init__(self, parent=None):

        super(CustWidget, self).__init__(parent)

        self.okButton = QPushButton("OK", self)

        #使用setObjectName设置对象名称self.okButton.setObjectName("okButton")

        layout = QHBoxLayout()

        layout.addWidget(self.okButton)

        self.setLayout(layout)

        QtCore.QMetaObject.connectSlotsByName(self)

    @QtCore.pyqtSlot()

    def on_okButton_clicked(self):

        print("单击了OK按钮")if__name__=="__main__":

    app =  QApplication(sys.argv)

    win = CustWidget()

    win.setWindowTitle('装饰器信号和槽')

    win.setGeometry(300,300,600,400)

    win.show()

    app.exec_()

   运行脚本,显示效果如图所示。单击“OK”按钮,控制台打印出预期的调试信息。

图4

   有的读者可能注意到,我们一直没有解释下面这行代码的含义:QtCore.QMetaObject.connectSlotsByName(QObject),事实上,它是在PyQt 5中根据信号名称自动连接到槽函数的核心代码。通过前面章节中的例子可以知道,使用pyuic5命令生成的代码中会带有这么一行代码,接下来对其进行解释。

  这行代码用来将QObject中的子孙对象的某些信号按照其objectName连接到相应的槽函数。这句话读起来有些拗口,这里举个例子进行简单说明。以上面例子中的代码为例:

  假设代码QtCore.QMetaObject.connectSlotsByName(self)已经执行,则下面的代码:

@QtCore.pyqtSlot()    def on_okButton_clicked(self):

    print("单击了OK按钮")


  会被自动识别为下面的代码(注意,函数中去掉了on,因为on会受到connectSlotsByName的影响,加上on运行时会出现问题):

def__init__(self, parent=None):

    self.okButton.clicked.connect(self.okButton_clicked)

    def okButton_clicked(self):

        print("单击了OK按钮")

 实例如下:

 View Code

运行上述代码,结果和图4一样。

六、信号与槽的断开和连接

  有时候基于某些原因,想要临时或永久断开某个信号与槽的连接。这就是本节案例想要达到的目的。其完整代码如下:

fromPyQt5.QtCoreimport QObject , pyqtSignalclass SignalClass(QObject):

    # 声明无参数的信号signal1 = pyqtSignal()

    # 声明带一个int类型参数的信号signal2 = pyqtSignal(int)

    def__init__(self,parent=None):

        super(SignalClass,self).__init__(parent)

        # 将信号signal1连接到sin1Call和sin2Call这两个槽函数        self.signal1.connect(self.sin1Call)

        self.signal1.connect(self.sin2Call)

        # 将信号signal2连接到信号signal1        self.signal2.connect(self.signal1)

        # 发射信号        self.signal1.emit()

        self.signal2.emit(1)

        # 断开signal1、signal2信号与各槽函数的连接        self.signal1.disconnect(self.sin1Call)

        self.signal1.disconnect(self.sin2Call)

        self.signal2.disconnect(self.signal1)

        # 将信号signal1和signal2连接到同一个槽函数sin1Call        self.signal1.connect(self.sin1Call)

        self.signal2.connect(self.sin1Call)

        # 再次发射信号        self.signal1.emit()

        self.signal2.emit(1)

    def sin1Call(self):

        print("signal-1 emit")

    def sin2Call(self):

        print("signal-2 emit")if__name__=='__main__':

    signal = SignalClass()

  运行结果如下:

signal-1 emit

signal-2 emit

signal-1 emit

signal-2 emit

signal-1 emit

signal-1 emit

七、多线程中信号与槽的使用

1、简单多线程中信号与槽的使用 

  最简单的多线程使用方法是利用QThread函数,如下代码展示了QThread函数和信号与槽简单的结合方法。其完整代码如下: 

# 多线程中信号与槽的使用fromPyQt5.QtWidgetsimport  QApplication ,QWidgetfromPyQt5.QtCoreimport QThread ,  pyqtSignalimport sysclass Main(QWidget):

    def__init__(self, parent = None):

        super(Main,self).__init__(parent)

        # 创建一个线程实例并设置名称、变量、信号与槽self.thread = MyThread()       

        self.thread.setIdentity("thread1")

        self.thread.sinOut.connect(self.outText)

        self.thread.setVal(6)

    def outText(self,text):

        print(text)class MyThread(QThread):

    sinOut = pyqtSignal(str)

    def__init__(self,parent=None):

        super(MyThread,self).__init__(parent)

        self.identity = None

    def setIdentity(self,text):

        self.identity = text

    def setVal(self,val):

        self.times = int(val)

        # 执行线程的run方法        self.start()

    def run(self):

        whileself.times > 0and self.identity:

            # 发射信号self.sinOut.emit(self.identity+"==>"+str(self.times))

            self.times -= 1if__name__=='__main__':

    app = QApplication(sys.argv)

    main = Main()

    main.show()

    sys.exit(app.exec_())

运行结果:

thread1==>6thread1==>5thread1==>4thread1==>3thread1==>2thread1==>1


2、多线程处理显示和逻辑运算分开

  有时在开发程序时经常会执行一些耗时的操作,这样就会导致界面卡顿,这也是多线程的应用范围之一——为了解决这个问题,我们可以创建多线程,使用主线程更新界面,使用子线程实时处理数据,最后将结果显示到界面上。

  下例中,定义了一个后台线程类BackendThread来模拟后台耗时操作,在这个线程类中定义了信号update_date。使用BackendThread线程类在后台处理数据,每秒发射一次自定义信号update_date。

  在初始化窗口界面时,定义后台线程类BackendThread,并把线程类的信号update_date连接到槽函数handleDisplay()。这样后台线程每发射一次信号,就可以把最新的时间值实时显示在前台窗口的QLineEdit文本对话框中,完整示例代码如下:

fromPyQt5.QtCoreimport QThread ,  pyqtSignal,  QDateTimefromPyQt5.QtWidgetsimport QApplication,  QDialog,  QLineEditimport timeimport sysclassBackendThread(QThread):# 该类模拟后台# 通过类成员对象定义信号update_date = pyqtSignal(str)

    # 处理业务逻辑def run(self):

        while True:

            data = QDateTime.currentDateTime()

            currTime = data.toString("yyyy-MM-dd hh:mm:ss")

            self.update_date.emit(str(currTime))  #通过sleep(1),每秒发射一个信号time.sleep(1)# class Window(QDialog):  #界面类,用于显示class Window(PQW.QWidget):

    def__init__(self):

        # QDialog.__init__(self)super().__init__()

        self.setWindowTitle('PyQt 5界面实时更新例子')

        self.resize(400, 100)

        self.input = QLineEdit(self)

        self.input.resize(400, 30)

        self.initUI()

    def initUI(self):

        # 创建线程self.backend = BackendThread()

        # 连接信号        self.backend.update_date.connect(self.handleDisplay)

        # 开始线程        self.backend.start()

    # 将当前时间输出到文本框def handleDisplay(self, data):

        self.input.setText(data)if__name__=='__main__':

    app = QApplication(sys.argv)

    win = Window()

    win.show()

    sys.exit(app.exec_())

 运行结果:


图5

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容

  • 1、概述 信号槽是 Qt 框架引以为豪的机制之一。所谓信号槽,实际就是观察者模式。当某个事件发生之后,比如,按钮检...
    你的社交帐号昵阅读 45,282评论 0 9
  • 这个例子相对综合一些,包括qt的布局,实现无边框效果,无边框也就是没有了窗口的title栏,没有title栏就不能...
    用电热毯烤猪阅读 3,073评论 0 50
  • 信号和槽(Signals and Slots) Qt库第一个认识到在几乎所有情况下,程序员不需要或甚至不想知道所有...
    珞珈村下山阅读 9,823评论 0 23
  • 很久木有画画了,练习下素描,第一次用8开纸画,很多细节的,没有耐心去一点一点画了,手都黑了…… 忙工作忙了很长一段...
    小青童画阅读 1,175评论 15 25
  • 奋战春考,医学院医学检验专业!
    s时尚阅读 180评论 0 0