首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >如何断言PyQt5信号的身份?

如何断言PyQt5信号的身份?
EN

Stack Overflow用户
提问于 2019-02-15 15:14:44
回答 2查看 831关注 0票数 1

我所处的情况是,我想断言PyQt5信号的身份。具体来说,我想检查来自给定的clicked对象的QPushButton信号是否与存储在名为signal的变量中的信号相同。下面的片段说明了这种情况:

代码语言:javascript
复制
from PyQt5.QtWidgets import QApplication, QPushButton

app = QApplication([])
widget = QPushButton()
signal = widget.clicked

widget.clicked == signal
Out[1]: False

widget.clicked is signal
Out[2]: False

id(widget.clicked) == id(signal)
Out[3]: False

如图所示,分别涉及==isid()的三种比较都产生了False,即它们无法断言左、右参数的同一性。

有什么方法可以断言widget.clickedsignal引用的是相同的信号?

EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2019-02-15 17:25:44

每次调用它时,都会创建作为实体的信号,因为它们代表着不同的连接:

代码语言:javascript
复制
In [1]: import sys

In [2]: from PyQt5 import QtWidgets

In [3]: app = QtWidgets.QApplication(sys.argv)

In [4]: button = QtWidgets.QPushButton()

In [5]: id(button.clicked)
Out[5]: 140150155639464

In [6]: id(button.clicked)
Out[6]: 140150154507528

In [7]: id(button.clicked)
Out[7]: 140150155640184

In [8]: id(button.clicked)
Out[8]: 140150155640504

In [9]: id(button.clicked)
Out[9]: 140150154510128

In [10]: id(button.clicked)
Out[10]: 140149427454320

因此,如果在同一信号和时隙之间连接100次,并且发出信号时,该插槽将被调用100次:

代码语言:javascript
复制
import sys
from PyQt5 import QtCore, QtWidgets

def foo():
    print("clicked")

if __name__ == '__main__':

    app = QtWidgets.QApplication(sys.argv)
    button = QtWidgets.QPushButton("Press me")
    button.show()
    for _ in range(10):
        button.clicked.connect(foo)
    # the click is emulated
    QtCore.QTimer.singleShot(1000, lambda: button.animateClick(500))
    QtCore.QTimer.singleShot(2000, app.quit)
    sys.exit(app.exec_())

输出:

代码语言:javascript
复制
clicked
clicked
clicked
clicked
clicked
clicked
clicked
clicked
clicked
clicked

因此,不可能直接解决您的问题,但我认为您的目标是区分哪个对象发出了调用时隙的信号,因为您可能有几个对象连接到同一个插槽,如果有解决方案的话:

1.使用sender()方法

如果插槽属于QObject (或从QObject派生的类作为小部件),则可以使用发送方方法获取发出信号的对象。

代码语言:javascript
复制
import sys
from PyQt5 import QtCore, QtWidgets

class Widget(QtWidgets.QWidget):
    def __init__(self, parent=None):
        super(Widget, self).__init__(parent)

        self.button_1 = QtWidgets.QPushButton("button 1")
        self.button_1.clicked.connect(self.foo)
        self.button_2 = QtWidgets.QPushButton("button 2")
        self.button_2.clicked.connect(self.foo)
        self.button_3 = QtWidgets.QPushButton("button 3")
        self.button_3.clicked.connect(self.foo)

        lay = QtWidgets.QVBoxLayout(self)
        lay.addWidget(self.button_1)
        lay.addWidget(self.button_2)
        lay.addWidget(self.button_3)

    @QtCore.pyqtSlot()
    def foo(self):
        button = self.sender()
        if button is self.button_1:
            print("button_1")
        elif button is self.button_2:
            print("button_2")
        elif button is self.button_3:
            print("button_3")

if __name__ == '__main__':

    app = QtWidgets.QApplication(sys.argv)
    w = Widget()
    w.show()
    sys.exit(app.exec_())

2.将对象作为附加参数传递

2.1 lambda函数

代码语言:javascript
复制
import sys
from PyQt5 import QtCore, QtWidgets

class Widget(QtWidgets.QWidget):
    def __init__(self, parent=None):
        super(Widget, self).__init__(parent)

        self.button_1 = QtWidgets.QPushButton("button 1")
        self.button_1.clicked.connect(lambda *args, b=self.button_1 : self.foo(b))
        self.button_2 = QtWidgets.QPushButton("button 2")
        self.button_2.clicked.connect(lambda *args, b=self.button_2 : self.foo(b))
        self.button_3 = QtWidgets.QPushButton("button 3")
        self.button_3.clicked.connect(lambda *args, b=self.button_3 : self.foo(b))

        lay = QtWidgets.QVBoxLayout(self)
        lay.addWidget(self.button_1)
        lay.addWidget(self.button_2)
        lay.addWidget(self.button_3)

    def foo(self, button):
        if button is self.button_1:
            print("button_1")
        elif button is self.button_2:
            print("button_2")
        elif button is self.button_3:
            print("button_3")

if __name__ == '__main__':

    app = QtWidgets.QApplication(sys.argv)
    w = Widget()
    w.show()
    sys.exit(app.exec_())

2.1 functools.partial函数

代码语言:javascript
复制
import sys
from PyQt5 import QtCore, QtWidgets
from functools import partial

class Widget(QtWidgets.QWidget):
    def __init__(self, parent=None):
        super(Widget, self).__init__(parent)

        self.button_1 = QtWidgets.QPushButton("button 1")
        self.button_1.clicked.connect(partial(self.foo, self.button_1))
        self.button_2 = QtWidgets.QPushButton("button 2")
        self.button_2.clicked.connect(partial(self.foo, self.button_2))
        self.button_3 = QtWidgets.QPushButton("button 3")
        self.button_3.clicked.connect(partial(self.foo, self.button_3))

        lay = QtWidgets.QVBoxLayout(self)
        lay.addWidget(self.button_1)
        lay.addWidget(self.button_2)
        lay.addWidget(self.button_3)

    def foo(self, button):
        if button is self.button_1:
            print("button_1")
        elif button is self.button_2:
            print("button_2")
        elif button is self.button_3:
            print("button_3")

if __name__ == '__main__':

    app = QtWidgets.QApplication(sys.argv)
    w = Widget()
    w.show()
    sys.exit(app.exec_())

在我的例子中,如果我可以使用发件人,那么我更愿意使用它,然后是functools.partial,最后是lambda方法。

票数 2
EN

Stack Overflow用户

发布于 2019-02-15 16:23:19

试一试:

代码语言:javascript
复制
import sys
from PyQt5.QtWidgets import QApplication, QPushButton

app = QApplication([])

def clickButton(w):
    # Check what you clicked here.                        # <-----
    return print(w.text())

widget  = QPushButton('Button 1')
widget.clicked.connect(lambda : clickButton(widget))

widget2 = QPushButton('Button 2')
widget2.clicked.connect(lambda : clickButton(widget2))

widget.setGeometry(300, 150, 100, 100)
widget.show()

widget2.setGeometry(450, 150, 100, 100)
widget2.show()

sys.exit(app.exec_())

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/54712203

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档