如何在PyQt中通过按钮点击向函数传递参数?

67 投票
7 回答
147432 浏览
提问于 2025-04-16 22:02

我想在点击按钮的时候,把一些参数传递给一个函数。我应该在这一行 button.connect(button, QtCore.SIGNAL('clicked()'), calluser(name)) 加上什么,才能把值传给这个函数呢?

def calluser(name):
    print name

def Qbutton():
    button = QtGui.QPushButton("button",widget)
    name = "user"
    button.setGeometry(100,100, 60, 35)
    button.connect(button, QtCore.SIGNAL('clicked()'), calluser(name))

还有一件事,按钮是通过 for 循环生成的,所以 name 的值会变化。我想把每个名字和按钮关联起来。我在 Pytk 中也做过类似的事情,通过 for 循环来调用带参数的函数,当按钮被点击时。

7 个回答

41

通常,图形用户界面(GUI)是通过类来构建的。通过使用绑定的方法作为回调(就像下面的 self.calluser),你可以通过 self 的属性(比如 self.name)来“传递”信息给回调函数:

例如,使用稍微修改过的代码来自 这个教程

import sys
import PyQt4.QtCore as QtCore
import PyQt4.QtGui as QtGui

class QButton(QtGui.QWidget):
    def __init__(self, parent=None):
        QtGui.QWidget.__init__(self, parent)
        self.button = QtGui.QPushButton('Button', self)
        self.name='me'
        self.button.clicked.connect(self.calluser)
    def calluser(self):
        print(self.name)

def demo_QButton():
    app = QtGui.QApplication(sys.argv)
    tb = QButton()
    tb.show()
    app.exec_()

if __name__=='__main__':
    demo_QButton()

因为回调函数本身总是没有额外的参数,所以当你需要向多个回调传递不同的信息时,就需要为每个按钮创建不同的回调函数。

由于这样做可能会很麻烦(如果手动完成的话),所以可以使用一个函数工厂。下面有一个例子。函数工厂是一个闭包,它可以接收额外的参数,而内部函数在被调用时可以访问这些参数:

class ButtonBlock(QtGui.QWidget):

    def __init__(self, *args):
        super(QtGui.QWidget, self).__init__()
        grid = QtGui.QGridLayout()
        names = ('One', 'Two', 'Three', 'Four', 'Five',
                 'Six', 'Seven', 'Eight', 'Nine', 'Ten')
        for i, name in enumerate(names):
            button = QtGui.QPushButton(name, self)
            button.clicked.connect(self.make_calluser(name))
            row, col = divmod(i, 5)
            grid.addWidget(button, row, col)
        self.setLayout(grid)

    def make_calluser(self, name):
        def calluser():
            print(name)
        return calluser

app = QtGui.QApplication(sys.argv)
tb = ButtonBlock()
tb.show()
app.exec_()
42

我尝试了一种高效的方法,效果很好。你可以使用下面的代码:

from functools import partial

def calluser(name):
    print name

def Qbutton():
    button = QtGui.QPushButton("button",widget)
    name = "user"
    button.setGeometry(100,100, 60, 35)
    button.clicked.connect(partial(calluser,name))
123

你可以简单地写

name = "user"
button.clicked.connect(lambda: calluser(name))

撰写回答