从函数中按名称访问实例变量

时间:2017-02-25 12:44:09

标签: python-3.x pyqt5

我需要为一个类创建一个UI,该类通过一组变量进行大量计算。代码大致如下所示:

class MainWindow:
    def __init__(self):
        pulse = Pulse()

        self.var1_lineEdit.textEdited.connect(self.set_attribute(pulse, "var1", new_value))
        self.var2_lineEdit.textEdited.connect(self.set_attribute(pulse, "var2", new_value))
        self.var3_lineEdit.textEdited.connect(self.set_attribute(pulse, "var3", new_value))
        ....

    def set_attribute(pulse, var_name, value):
        # get value from line edit
        # update value in pulse
        # update other calculations

class Pulse:
    def __init__(self):
        var1 = 1
        var2 = 2
        var3 = 3
        ....

我的问题是如何编写set_attribute代码。有没有办法通过将名称作为函数参数传递来访问另一个类的各个变量?我想避免为每个变量定义一个新函数。

或者有更简单的方法吗?我不太熟悉OOP和编写UI的正确方法。

1 个答案:

答案 0 :(得分:1)

修改

您提出的信号/插槽连接会让您遇到麻烦。使用connect函数将信号连接到插槽,该函数接受python callable作为其参数 - 而不是函数调用本身。这是

之间的区别
widget.textChanged.connect(self.set_attribute) # right syntax
widget.textChanged.connect(self.set_attribute()) # wrong syntax

那么,你如何告诉set_attribute该做什么? ekhumoro告诉我我的方式错误:使用QSignalMapper。这是一种过时的技术。最好只使用lambda函数(或partials)来获得所需的效果。

我更新了我的演示以使用lambdas

出于本演示的目的,我创建了一个简单的QMainWindow,其中包含两个QLineEdit小部件和一个QPushButton

# file ui_main.py
from PyQt5 import QtCore, QtGui, QtWidgets

class Ui_MainWindow(object):
    def setupUi(self, MainWindow):
        MainWindow.setObjectName("MainWindow")
        MainWindow.resize(359, 249)
        self.centralwidget = QtWidgets.QWidget(MainWindow)
        self.centralwidget.setObjectName("centralwidget")
        self.formLayoutWidget = QtWidgets.QWidget(self.centralwidget)
        self.formLayoutWidget.setGeometry(QtCore.QRect(0, 0, 351, 51))
        self.formLayoutWidget.setObjectName("formLayoutWidget")
        self.formLayout = QtWidgets.QFormLayout(self.formLayoutWidget)
        self.formLayout.setObjectName("formLayout")
        self.configParam1Label = QtWidgets.QLabel(self.formLayoutWidget)
        self.configParam1Label.setObjectName("configParam1Label")
        self.formLayout.setWidget(0, QtWidgets.QFormLayout.LabelRole, self.configParam1Label)
        self.configEntry = QtWidgets.QLineEdit(self.formLayoutWidget)
        self.configEntry.setObjectName("configEntry")
        self.formLayout.setWidget(0, QtWidgets.QFormLayout.FieldRole, self.configEntry)
        self.configParam2Label = QtWidgets.QLabel(self.formLayoutWidget)
        self.configParam2Label.setObjectName("configParam2Label")
        self.formLayout.setWidget(1, QtWidgets.QFormLayout.LabelRole, self.configParam2Label)
        self.configEntry2 = QtWidgets.QLineEdit(self.formLayoutWidget)
        self.configEntry2.setObjectName("configEntry2")
        self.formLayout.setWidget(1, QtWidgets.QFormLayout.FieldRole, self.configEntry2)
        self.pulseButton = QtWidgets.QPushButton(self.centralwidget)
        self.pulseButton.setGeometry(QtCore.QRect(130, 140, 75, 23))
        self.pulseButton.setObjectName("pulseButton")
        MainWindow.setCentralWidget(self.centralwidget)

        self.retranslateUi(MainWindow)
        QtCore.QMetaObject.connectSlotsByName(MainWindow)

    def retranslateUi(self, MainWindow):
        _translate = QtCore.QCoreApplication.translate
        MainWindow.setWindowTitle(_translate("MainWindow", "MainWindow"))
        self.configParam1Label.setText(_translate("MainWindow", "Config Param 1"))
        self.configParam2Label.setText(_translate("MainWindow", "Config Param 2"))
        self.pulseButton.setText(_translate("MainWindow", "GetPulse"))

在另一个文件中,在一些导入之后,定义Pulse - 一个包含两个属性的简单类

# file main.py
from PyQt5.QtCore import (QSignalMapper, pyqtSlot)
from PyQt5.QtWidgets import (QApplication, QMainWindow, QMessageBox)

from ui_main import Ui_MainWindow

class Pulse:
    def __init__(self):
        self.param_one = None
        self.param_two = None

现在,定义MainWindow,设置其用户界面并为其提供Pulse

class MainWindow(QMainWindow):
    def __init__(self, parent=None):
        super(MainWindow, self).__init__(parent)

        self.pulse = Pulse()

        self.ui = Ui_MainWindow()
        self.ui.setupUi(self)

        self.widget_one = self.ui.configEntry
        self.widget_two = self.ui.configEntry2
        self.pulse_button = self.ui.pulseButton

        self.widget_one.textChanged.connect(lambda: self.widget_handler('param_one', self.widget_one.text()))
        self.widget_one.textChanged.connect(lambda: self.widget_handler('param_two', self.widget_two.text()))

    def widget_handler(self, attr, value):
        setattr(self.pulse,attr,value)

    def handler(self, idx):
        widget, attr = self.attribute_widget_list[idx]
        widget_value = widget.text()
        setattr(self.pulse,attr,widget_value)

setattr相当于,例如self.pulse.param_one = widget_value

为了证明它有效,我们使用QPushButton来探测Pulse的状态:

    @pyqtSlot()
    def on_pulseButton_clicked(self):
        message = QMessageBox()
        message.setText("{} - {}".format(self.pulse.param_one, self.pulse.param_two))
        message.exec()

这里我使用的是不同的信号/插槽连接方案,您可以阅读here

以下是我们运行脚本的方式:

if __name__ == '__main__':
    import sys

    app = QApplication(sys.argv)
    window = MainWindow()
    window.show()
    sys.exit(app.exec_())

希望有所帮助!