PyQt5零基础入门:信号与槽

前言

PyQt5的信号与槽是一种对象之间的通信机制,允许一个QObject对象发出信号,与之相连接的槽函数将会自动执行。信号与槽有以下几个特点:

  • 信号可以携带任意个、任意类型的参数。
  • 一个信号可以连接多个槽,即一个槽可以监听多个信号。
  • 信号与槽的连接方式可以是同步连接,也可以是异步连接。
  • 信号与槽可能会在不同的线程间连接,这时信号连接只能是异步方式。
  • 信号不是必须有槽和其连接,即信号可以不处理。
  • 信号个槽的连接可以被断开。

PyQt5使用pyqtSignal函数为QObject对象创建信号,并把信号定义为类的属性。使用connect()函数可以将信号绑定到槽函数上,使用disconnect()函数可以接触信号和槽之间的绑定。

信号与槽单对单

直接连接

python 复制代码
import sys
from PyQt5.QtWidgets import *

class Window(QWidget):
    def __init__(self):
        super(Window, self).__init__()
        self.btn = QPushButton("Start", self)
        self.btn.clicked.connect(self.change_text)

    def change_text(self):
        if self.btn.text() == "Start":
            self.btn.setText("Stop")
        else:
            self.btn.setText("Start")

if __name__ == "__main__":
    app = QApplication([])
    window = Window()
    window.show()
    sys.exit(app.exec())

在上述代码中创建一个QPushButton对象self.btn,并将其文本设置为"Start"。这个按钮被添加到当前窗口中(由self表示)。

self.btn.clicked.connect(self.change_text): 当按钮被点击时,将调用change_text方法。在change_text的方法中根据当前按钮显示的文字改变为另一个文字。

这个简单的程序创建了一个窗口,其中包含一个可以切换其标签的按钮("Start"和"Stop")。每次点击按钮时,都会调用change_text方法来更改其标签。

运行结果如下,运行时按钮上的文字是Start,按下之后会变成Stop

使用lambda表达式

python 复制代码
import sys
from PyQt5.QtWidgets import *

class Window(QWidget):
    def __init__(self):
        super().__init__()
        btn = QPushButton("Start", self)
        btn.clicked.connect(lambda:btn.setText("Stop"))


if __name__ == "__main__":
    app = QApplication([])
    win = Window()
    win.show()
    sys.exit(app.exec())

在上述代码中我们并没有创建新的槽函数,只采用lambda表达式来设置按钮的值。不过与上一个方法相比,这个代码只能将Start变成Stop,但是不能变回Start

运行结果与上种方式相同

信号与槽多对多

一个信号连接多个槽

python 复制代码
import sys
from PyQt5.QtWidgets import *

class Window(QWidget):
    def __init__(self):
        super(Window, self).__init__()
        self.btn = QPushButton("Start", self)
        self.btn.clicked.connect(self.change_text)
        self.btn.clicked.connect(self.change_size)

    def change_text(self):
        if self.btn.text() == "Start":
            self.btn.setText("Stop")
        else:
            self.btn.setText("Start")

    def change_size(self):
        self.btn.resize(150, 30)


if __name__ == "__main__":
    app = QApplication([])
    win = Window()
    win.show()
    sys.exit(app.exec())

在代码中,按钮除了连接到了改变按钮文字的槽函数之外还连接到了更换按钮大小的槽函数,所以在按下按钮后除了改变了文字,还改变了按钮大小。

运行结果如下

多个信号连接一个槽

python 复制代码
import sys
from PyQt5.QtWidgets import *

class Window(QWidget):
    def __init__(self):
        super(Window, self).__init__()
        self.btn = QPushButton("Start", self)
        self.btn.pressed.connect(self.change_text)
        self.btn.released.connect(self.change_text)

    def change_text(self):
        if self.btn.text() == "Start":
            self.btn.setText("Stop")
        else:
            self.btn.setText("Start")


if __name__ == "__main__":
    app = QApplication([])
    win = Window()
    win.show()
    sys.exit(app.exec())

在上述代码中

当按钮被按下时,会触发 pressed 信号,连接 pressed 信号到 self.change_text 函数,每次按钮被按下时,self.change_text 函数都会被调用。

当按钮被释放时,会触发 released 信号,连接 released 信号到 self.change_text 函数,每次按钮被释放时,self.change_text 函数都会被调用。

所以,self.change_text 函数会在按钮被按下和释放时都被调用。

信号与信号连接

python 复制代码
import sys
from PyQt5.QtWidgets import *

class Window(QWidget):
    def __init__(self):
        super(Window, self).__init__()
        self.btn = QPushButton("Start", self)
        self.btn.pressed.connect(self.btn.released)
        self.btn.released.connect(self.change_text)

    def change_text(self):
        if self.btn.text() == "Start":
            self.btn.setText("Stop")
        else:
            self.btn.setText("Start")


if __name__ == "__main__":
    app = QApplication([])
    win = Window()
    win.show()
    sys.exit(app.exec())

在上述代码中将pressed连接到released,而released连接到了change_text。运行效果和上面代码一致

自定义信号

python 复制代码
import sys
from PyQt5.QtCore import *
from PyQt5.QtWidgets import *


class Window(QWidget):
    my_signal = pyqtSignal(int, int)
    def __init__(self):
        super().__init__()
        self.my_signal.connect(self.my_slot)

    def my_slot(self, x, y):
        print(x)
        print(y)

    def mousePressEvent(self, event):
        x = event.pos().x()
        y = event.pos().y()
        self.my_signal.emit(x, y)


if __name__ == "__main__":
    app = QApplication([])
    win = Window()
    win.show()
    sys.exit(app.exec())

在PyQt5中mousePressEvent是鼠标按下时会触发的信号,我们自己定义的信号self.my_signal与槽函数my_slot相连接,当鼠标按下时,self.my_signal使用emit触发self.my_signal信号。效果就是点击窗口会输出当前鼠标位置。

相关推荐
测试老哥41 分钟前
python+requests+excel 接口测试
自动化测试·软件测试·python·测试工具·测试用例·excel·接口测试
AI纪元故事会42 分钟前
冰泪与雨丝:一个AI的Python挽歌
开发语言·人工智能·python
ColderYY1 小时前
Python连接MySQL数据库
数据库·python·mysql
寒秋丶1 小时前
Milvus:数据库层操作详解(二)
数据库·人工智能·python·ai·ai编程·milvus·向量数据库
凯歌的博客2 小时前
python虚拟环境应用
linux·开发语言·python
西柚小萌新2 小时前
【深入浅出PyTorch】--8.1.PyTorch生态--torchvision
人工智能·pytorch·python
MonkeyKing_sunyuhua2 小时前
can‘t read /etc/apt/sources.list: No such file or directory
python
多喝开水少熬夜3 小时前
损失函数系列:focal-Dice-vgg
图像处理·python·算法·大模型·llm
初学小刘4 小时前
基于 U-Net 的医学图像分割
python·opencv·计算机视觉
B站计算机毕业设计之家4 小时前
Python招聘数据分析可视化系统 Boss直聘数据 selenium爬虫 Flask框架 数据清洗(附源码)✅
爬虫·python·selenium·机器学习·数据分析·flask