PYQT5開(kāi)啟多個(gè)線程和窗口,多線程與多窗口的交互實(shí)例
每點(diǎn)擊一次按鈕,彈出一個(gè)對(duì)話框(子窗口),同時(shí)開(kāi)啟一個(gè)子線程來(lái)執(zhí)行任務(wù)并更新對(duì)話框內(nèi)容,關(guān)閉對(duì)話框則關(guān)閉對(duì)應(yīng)子線程
1. 建立一個(gè)簡(jiǎn)單的主界面和一個(gè)自定義對(duì)話框

from PyQt5 import QtCore, QtGui, QtWidgets
class Ui_MainWindow(object):
def setupUi(self, MainWindow):
MainWindow.setObjectName("MainWindow")
MainWindow.resize(327, 303)
self.centralwidget = QtWidgets.QWidget(MainWindow)
self.centralwidget.setObjectName("centralwidget")
self.gridLayout = QtWidgets.QGridLayout(self.centralwidget)
self.gridLayout.setObjectName("gridLayout")
spacerItem = QtWidgets.QSpacerItem(40, 20, QtWidgets.QSizePolicy.Expanding, QtWidgets.QSizePolicy.Minimum)
self.gridLayout.addItem(spacerItem, 0, 0, 1, 1)
self.pushButton = QtWidgets.QPushButton(self.centralwidget)
self.pushButton.setObjectName("pushButton")
self.gridLayout.addWidget(self.pushButton, 0, 1, 1, 1)
spacerItem1 = QtWidgets.QSpacerItem(40, 20, QtWidgets.QSizePolicy.Expanding, QtWidgets.QSizePolicy.Minimum)
self.gridLayout.addItem(spacerItem1, 0, 2, 1, 1)
MainWindow.setCentralWidget(self.centralwidget)
self.menubar = QtWidgets.QMenuBar(MainWindow)
self.menubar.setGeometry(QtCore.QRect(0, 0, 327, 23))
self.menubar.setObjectName("menubar")
MainWindow.setMenuBar(self.menubar)
self.statusbar = QtWidgets.QStatusBar(MainWindow)
self.statusbar.setObjectName("statusbar")
MainWindow.setStatusBar(self.statusbar)
self.retranslateUi(MainWindow)
self.pushButton.clicked.connect(MainWindow.open_dialog)
QtCore.QMetaObject.connectSlotsByName(MainWindow)
def retranslateUi(self, MainWindow):
_translate = QtCore.QCoreApplication.translate
MainWindow.setWindowTitle(_translate("MainWindow", "MainWindow"))
self.pushButton.setText(_translate("MainWindow", "多線程彈窗"))
class Ui_Dialog(object):
def setupUi(self, Dialog):
Dialog.setObjectName("Dialog")
Dialog.resize(369, 128)
self.gridLayout = QtWidgets.QGridLayout(Dialog)
self.gridLayout.setObjectName("gridLayout")
self.buttonBox = QtWidgets.QDialogButtonBox(Dialog)
self.buttonBox.setOrientation(QtCore.Qt.Horizontal)
self.buttonBox.setStandardButtons(QtWidgets.QDialogButtonBox.Cancel|QtWidgets.QDialogButtonBox.Ok)
self.buttonBox.setObjectName("buttonBox")
self.gridLayout.addWidget(self.buttonBox, 1, 0, 1, 1)
self.progressBar = QtWidgets.QProgressBar(Dialog)
self.progressBar.setProperty("value", 24)
self.progressBar.setObjectName("progressBar")
self.gridLayout.addWidget(self.progressBar, 0, 0, 1, 1)
self.retranslateUi(Dialog)
self.buttonBox.accepted.connect(Dialog.accept)
self.buttonBox.rejected.connect(Dialog.reject)
QtCore.QMetaObject.connectSlotsByName(Dialog)
def retranslateUi(self, Dialog):
_translate = QtCore.QCoreApplication.translate
Dialog.setWindowTitle(_translate("Dialog", "Dialog"))
2. 每點(diǎn)擊一次按鈕,打開(kāi)一個(gè)彈窗
class DialogWindow(QDialog, Ui_Dialog): def __init__(self, parent=None): super(DialogWindow, self).__init__(parent) self.setupUi(self) class MainWindow(QtWidgets.QMainWindow, Ui_MainWindow): def __init__(self, parent=None): super(MainWindow, self).__init__(parent) self.setupUi(self) def open_dialog(self): dialog = DialogWindow(self) dialog.show() if __name__ == "__main__": app = QtWidgets.QApplication(sys.argv) mainWindow = MainWindow() mainWindow.show() sys.exit(app.exec_())
3. 打開(kāi)彈窗的同時(shí),打開(kāi)一個(gè)子線程,更新對(duì)話框中的進(jìn)度條

在子線程定義信號(hào),關(guān)聯(lián)對(duì)話框更新進(jìn)度條的槽函數(shù)
class DialogWindow(QDialog, Ui_Dialog):
def __init__(self, parent=None):
super(DialogWindow, self).__init__(parent)
self.setupUi(self)
def update_progressbar(self, p_int):
self.progressBar.setValue(p_int) # 更新進(jìn)度條
class MainWindow(QtWidgets.QMainWindow, Ui_MainWindow):
def __init__(self, parent=None):
super(MainWindow, self).__init__(parent)
self.setupUi(self)
self.count = 0
def open_dialog(self):
dialog = DialogWindow(self)
dialog.show()
self.thread = RunThread(self.count)
self.count += 1
self.thread.update_pb.connect(dialog.update_progressbar) # 關(guān)聯(lián)
self.thread.start()
class RunThread(QThread):
update_pb = pyqtSignal(int) # 定義更新進(jìn)度條的信號(hào)
def __init__(self, count):
super().__init__()
self.count = count
def run(self):
for i in range(100):
print('thread%s' % self.count, i, QThread().currentThreadId())
self.update_pb.emit(i)
time.sleep(1)
pass
if __name__ == "__main__":
app = QtWidgets.QApplication(sys.argv)
mainWindow = MainWindow()
mainWindow.show()
sys.exit(app.exec_())
4. 關(guān)閉對(duì)話框,則關(guān)閉對(duì)應(yīng)子線程
在對(duì)話框中添加自定義信號(hào),并重寫(xiě)關(guān)閉事件,在關(guān)閉窗口時(shí)發(fā)送關(guān)閉子線程的信號(hào)
class DialogWindow(QDialog, Ui_Dialog):
stop_thread = pyqtSignal() # 定義關(guān)閉子線程的信號(hào)
def __init__(self, parent=None):
super(DialogWindow, self).__init__(parent)
self.setupUi(self)
def update_progressbar(self, p_int):
self.progressBar.setValue(p_int)
def closeEvent(self, event):
self.stop_thread.emit()
pass
class MainWindow(QtWidgets.QMainWindow, Ui_MainWindow):
def __init__(self, parent=None):
super(MainWindow, self).__init__(parent)
self.setupUi(self)
self.count = 0
def open_dialog(self):
dialog = DialogWindow(self)
dialog.show()
self.thread = RunThread(self.count)
self.count += 1
self.thread.update_pb.connect(dialog.update_progressbar)
dialog.stop_thread.connect(self.thread.terminate)
self.thread.start()
class RunThread(QThread):
update_pb = pyqtSignal(int)
def __init__(self, count):
super().__init__()
self.count = count
def run(self):
for i in range(1, 101):
print('thread_%s' % self.count, i, QThread().currentThreadId())
self.update_pb.emit(i)
time.sleep(1)
pass
if __name__ == "__main__":
app = QtWidgets.QApplication(sys.argv)
mainWindow = MainWindow()
mainWindow.show()
sys.exit(app.exec_())
5. 使用線程池QThreadPool管理子線程
使用QThreadPool, 線程需要繼承QRunnable,而QRunnable只是namespace,沒(méi)有繼承QT的信號(hào)機(jī)制,
所以需要另外繼承QObject來(lái)使用信號(hào),我這里直接在線程中使用封裝的信號(hào)向外部傳遞信息
class DialogWindow(QDialog, Ui_Dialog):
stop_thread = pyqtSignal() # 定義關(guān)閉子線程的信號(hào)
def __init__(self, parent=None):
super(DialogWindow, self).__init__(parent)
self.setupUi(self)
def update_progressbar(self, p_int):
self.progressBar.setValue(p_int)
def closeEvent(self, event):
self.stop_thread.emit()
pass
class MainWindow(QtWidgets.QMainWindow, Ui_MainWindow):
def __init__(self, parent=None):
super(MainWindow, self).__init__(parent)
self.setupUi(self)
self.count = 0
self.pool = QThreadPool()
self.pool.globalInstance()
self.pool.setMaxThreadCount(10) # 設(shè)置最大線程數(shù)
def open_dialog(self):
dialog = DialogWindow(self)
dialog.show()
thread = RunThread(self.count)
self.count += 1
thread.signal.update_pb.connect(dialog.update_progressbar)
# dialog.stop_thread.connect(thread.stop)
# self.thread.start()
self.pool.start(thread) # 線程池分配一個(gè)線程運(yùn)行該任務(wù)
class Signal(QObject):
update_pb = pyqtSignal(int)
class RunThread(QRunnable):
def __init__(self, count):
super().__init__()
self.count = count
self.signal = Signal() # 信號(hào)
def run(self):
for i in range(1, 101):
print('thread_%s' % self.count, i, QThread().currentThreadId())
self.signal.update_pb.emit(i)
time.sleep(1)
if __name__ == "__main__":
app = QtWidgets.QApplication(sys.argv)
mainWindow = MainWindow()
mainWindow.show()
sys.exit(app.exec_())
QThreadPool沒(méi)有釋放正在運(yùn)行的線程的方法
以上這篇PYQT5開(kāi)啟多個(gè)線程和窗口,多線程與多窗口的交互實(shí)例就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
- 詳解PyQt5中Thread多線程的使用
- Python Pyqt5多線程更新UI代碼實(shí)例(防止界面卡死)
- 詳解PyQt5 GUI 接收UDP數(shù)據(jù)并動(dòng)態(tài)繪圖的過(guò)程(多線程間信號(hào)傳遞)
- Pyqt5 實(shí)現(xiàn)多線程文件搜索的案例
- PyQt5多線程防卡死和多窗口用法的實(shí)現(xiàn)
- python GUI庫(kù)圖形界面開(kāi)發(fā)之PyQt5多線程中信號(hào)與槽的詳細(xì)使用方法與實(shí)例
- 利用PyQt中的QThread類(lèi)實(shí)現(xiàn)多線程
- PyQt5中多線程模塊QThread使用方法的實(shí)現(xiàn)
- PyQt 異步任務(wù)多線程的幾種方案示例詳解
相關(guān)文章
python使用pyecharts庫(kù)畫(huà)地圖數(shù)據(jù)可視化的實(shí)現(xiàn)
這篇文章主要介紹了python使用pyecharts庫(kù)畫(huà)地圖數(shù)據(jù)可視化的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-03-03
在Python 不同級(jí)目錄之間模塊的調(diào)用方法
今天小編就為大家分享一篇在Python 不同級(jí)目錄之間模塊的調(diào)用方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-01-01
django 多數(shù)據(jù)庫(kù)及分庫(kù)實(shí)現(xiàn)方式
這篇文章主要介紹了django 多數(shù)據(jù)庫(kù)及分庫(kù)實(shí)現(xiàn)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-04-04
Python3 獲取文件屬性的方式(時(shí)間、大小等)
這篇文章主要介紹了Python3 獲取文件屬性的方式(時(shí)間、大小等),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-03-03
python 使用 requests 模塊發(fā)送http請(qǐng)求 的方法
本文分步驟給大家介紹了python 使用 requests 模塊發(fā)送http請(qǐng)求 的方法,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2018-12-12
Python安裝后測(cè)試連接MySQL數(shù)據(jù)庫(kù)方式
這篇文章主要介紹了Python安裝后測(cè)試連接MySQL數(shù)據(jù)庫(kù)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-07-07
python+django+sql學(xué)生信息管理后臺(tái)開(kāi)發(fā)
這篇文章主要為大家詳細(xì)介紹了python+django+sql學(xué)生信息管理后臺(tái)開(kāi)發(fā),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-01-01
利用Python實(shí)現(xiàn)Picgo圖床工具
這篇文章主要介紹了如何利用Python實(shí)現(xiàn)Picgo圖床工具,PyPicGo?是一款圖床工具,是PicGo是Python版實(shí)現(xiàn),并支持各種插件自定義插件,目前PyPicGo自帶了gitee、github、SM.MS和七牛云圖傳,以及rename、notify和typora等插件,下面來(lái)看文章內(nèi)容介紹,需要的朋友可以參考一下2021-11-11

