经过两天浑浑噩噩的摸索,终于弄明白了两个类中的信号传递,为了不在一个类里面堆叠太多的的代码,将功能拆分成几个类甚至将这几个类拆分成几个文件,这样一方面方便维护,另一方面还方便日后做扩展,但是这几天就卡在了两个类之间怎么传递信号了,例如类A中的lineEditor的值发生了改变怎么让类B捕获这个信号并且将本类的lineEditor的内容改变呢?在一个类里面可能很容易实现,甚至都不用动脑.本例实现从一个主窗口中改变lineEditor的内容,子窗口捕获到主窗口的内容发生改变并将内容与主窗口的lineEditor同步,主要就是信号槽的使用,首先在子类创建信号槽,并且在子类的lineEditor发生改变时发送信号,父类捕获信号,并进行相应的动作
from PyQt5.QtCore import pyqtSignal
from PyQt5.QtWidgets import QWidget, QLineEdit, QVBoxLayout, QApplication, QFormLayout, QDialog
import sys
class sub(QDialog): #创建一个dialog,用作被调用类
changeValue = pyqtSignal() #创建槽信号
def __init__(self, parent=None):
super(sub, self).__init__(parent)
self.initUi()
def initUi(self):
self.setWindowTitle('子窗口')
self.lineEdit = QLineEdit()
self.formlayout = QFormLayout()
self.formlayout.addRow('输入', self.lineEdit)
self.lineEdit.textChanged.connect(self.setValue)
self.vbox = QVBoxLayout()
self.vbox.addLayout(self.formlayout)
self.setLayout(self.vbox)
self.resize(200, 200)
self.show()
def setValue(self):
self.changeValue.emit()
class mainwin(QWidget):
def __init__(self):
super(mainwin, self).__init__()
self.ex = sub(self) #实例化子类dialog,这一步一定要在self.initUi前面,不然initUi中不能调用没有实例化的changeValue这个槽信号
self.initUi()
def initUi(self):
self.setWindowTitle('主窗口')
self.lineEdit = QLineEdit()
self.formlayout = QFormLayout()
self.formlayout.addRow('输出', self.lineEdit)
self.ex.changeValue.connect(self.getValue) #调用sub类中的changeValue槽信号并绑定信号到getValue这个方法
self.vbox = QVBoxLayout()
self.vbox.addLayout(self.formlayout)
self.setLayout(self.vbox)
self.resize(400, 400)
self.show()
def getValue(self):
self.lineEdit.setText(self.ex.lineEdit.text()) #将sub的lineEdit中的内容传递给mainwin本类中的lineEdit中
if __name__ == "__main__":
app = QApplication(sys.argv)
myshow = mainwin()
sys.exit(app.exec_())
下例是带参数的信号传递
from PyQt5.QtCore import pyqtSignal
from PyQt5.QtWidgets import QWidget, QLineEdit, QVBoxLayout, QApplication, QFormLayout, QDialog
import sys
class sub(QDialog): #创建一个dialog,用作被调用类
changeValue = pyqtSignal(str) #创建槽信号,str指定接收参数为str类型(字符串类型)
def __init__(self, parent=None):
super(sub, self).__init__(parent)
self.initUi()
def initUi(self):
self.setWindowTitle('子窗口')
self.lineEdit = QLineEdit()
self.formlayout = QFormLayout()
self.formlayout.addRow('输入', self.lineEdit)
self.lineEdit.textChanged.connect(self.setValue)
self.vbox = QVBoxLayout()
self.vbox.addLayout(self.formlayout)
self.setLayout(self.vbox)
self.resize(200, 200)
self.show()
def setValue(self):
self.changeValue.emit(str(self.lineEdit.text())) #发送带参数的信号,信号为本类中的lineEdit.text(),即lineEdit的内容
class mainwin(QWidget):
def __init__(self):
super(mainwin, self).__init__()
self.ex = sub(self) #调用子类dialog,这一步一定要在self.initUi前面,不然initUi中不能调用没有实例化的changeValue这个槽信号
self.initUi()
def initUi(self):
self.setWindowTitle('主窗口')
self.lineEdit = QLineEdit()
self.formlayout = QFormLayout()
self.formlayout.addRow('输出', self.lineEdit)
self.ex.changeValue.connect(self.getValue) #调用sub类中的changeValue槽信号并绑定信号到getValue这个方法
self.vbox = QVBoxLayout()
self.vbox.addLayout(self.formlayout)
self.setLayout(self.vbox)
self.resize(400, 400)
self.show()
def getValue(self, val): #接收信号的函数,并且接收参数
self.lineEdit.setText(val)#将参数传递给本类lineEdit
# self.lineEdit.setText(self.ex.lineEdit.text()) #将sub的lineEdit中的内容传递给mainwin本类中的lineEdit中
if __name__ == "__main__":
app = QApplication(sys.argv)
myshow = mainwin()
sys.exit(app.exec_())
效果: