每点击一次按钮,弹出一个对话框(子窗口),同时开启一个子线程来执行任务并更新对话框内容,关闭对话框则关闭对应子线程

1. 建立一个简单的主界面和一个自定义对话框

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. 每点击一次按钮,打开一个弹窗

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. 打开弹窗的同时,打开一个子线程,更新对话框中的进度条

在子线程定义信号,关联对话框更新进度条的槽函数

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) # 更新进度条

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) # 关联

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(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. 关闭对话框,则关闭对应子线程

在对话框中添加自定义信号,并重写关闭事件,在关闭窗口时发送关闭子线程的信号

class DialogWindow(QDialog, Ui_Dialog):

stop_thread = pyqtSignal() # 定义关闭子线程的信号

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,没有继承QT的信号机制,

所以需要另外继承QObject来使用信号,我这里直接在线程中使用封装的信号向外部传递信息

class DialogWindow(QDialog, Ui_Dialog):

stop_thread = pyqtSignal() # 定义关闭子线程的信号

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) # 设置最大线程数

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) # 线程池分配一个线程运行该任务

class Signal(QObject):

update_pb = pyqtSignal(int)

class RunThread(QRunnable):

def __init__(self, count):

super().__init__()

self.count = count

self.signal = Signal() # 信号

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没有释放正在运行的线程的方法

以上这篇PYQT5开启多个线程和窗口,多线程与多窗口的交互实例就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持我们。

本文标题: PYQT5开启多个线程和窗口,多线程与多窗口的交互实例

本文地址: http://www.cppcns.com/jiaoben/python/292862.html

pyqt5多进程 python_PYQT5开启多个线程和窗口,多线程与多窗口的交互实例相关推荐

  1. 线程的常用方法——currentThread方法||在main方法中直接调用run()方法,没有开启新的线程,以在run方法中的当前线程就是main线程||启动子线程,子线程会调用run方法

    线程的常用方法--currentThread方法 Thread.currentThread()方法可以获得当前线程 Java 中的任何一段代码都是执行在某个线程当中的. 执行当前代码的线程就是当前线程 ...

  2. C++优雅地开启/暂停/停止线程——基于观察者模式

    在编程中经常需要创建子线程来执行耗时的任务,避免主线程卡死,提高程序性能.其中有很多业务场景需要我们可以自由控制子线程的开启/暂停/终止.比如音视频播放器,除了用来操作ui控件的GUI主线程外,还有音 ...

  3. Java开启多个线程,但资源老是全被一个线程抢了去,求解答

    Java,我学线程的时候写练习碰到的问题 写了三个线程当做窗口去"卖票",三个线程的优先级都是默认的5,按理来说这哥三抢到资源的概率应该是一样的,但运行结果好像差了那么亿点点(三个 ...

  4. JavaSE-多线程基础-基于多线程的计时器应用

    目录 一.进程与线程的含义与区别 1.含义: 2.区别: 二.线程的四种创建方式及使用 1.继承Thread类实现线程的创建 2.使用Runnable接口实现线程的创建 3.使用Callable接口实 ...

  5. JAVA:线程总结及多线程实现的两种方法

    JAVA:线程总结 目录 目录 JAVA:线程总结 JAVA:线程总结 01_多线程(多线程的引入)(了解) 02_多线程(多线程并行和并发的区别)(了解) 03_多线程(Java程序运行原理和JVM ...

  6. python捕捉线程错误_Pythonrequests多线程抓取出现HTTPConnectionPoolMaxretiresexceeded异常...

    问题: Python requests 多线程抓取 出现HTTPConnectionPool Max retires exceeded异常 描述: 主要代码如下:import threading im ...

  7. java 线程开销_多线程的线程开销

    多线程中两个必要的开销:线程的创建.上下文切换 创建线程: 创建线程使用是直接向系统申请资源的,对操作系统来说,创建一个线程的代价是十分昂贵的, 需要给它分配内存.列入调度,同时在线程切换的时候还要执 ...

  8. 【Java 多线程】多线程带来的的风险-线程安全、多线程五个经典案例

    日常开发中如果用到多线程编程,也一定会涉及到线程安全问题 线程安全这个问题就不太好理解 正因为如此,程序猿们才尝试发明出更多的编程模型来处理并发编程的任务 例如:多进程.多线程.actor.csp.a ...

  9. java runnable线程锁_多线程 java 同步 、锁 、 synchronized 、 Thread 、 Runnable

    线程 1 线程概述 1.1 什么是线程 v  线程是程序执行的一条路径, 一个进程中可以包含多条线程 v  一个应用程序可以理解成就是一个进程 v  多线程并发执行可以提高程序的效率, 可以同时完成多 ...

最新文章

  1. CentOS 7.6 下安装 MySQL8.0.13
  2. 干货分享:如何使用Kubernetes的Ingress API
  3. linux 常用命令02--文件属性 以及软硬链接
  4. 如何从需求文档中辨认客户(一)
  5. matlab的示波器保存figure图像
  6. 烽火MAS短信服务器配置相关
  7. objectC时间用法
  8. Linux下的虚拟机拷贝与快照生成
  9. vba 添加outlook 签名_利用VBA发送附件电子邮件
  10. python程序员自我介绍范文_程序员的自我介绍|史上最有趣的版本
  11. 在linux下将当前目录文件全部小写含目录名
  12. Rust : future库
  13. Python:罗马数字转整数
  14. 4G DTU和4G工业路由器有哪些区别?
  15. 数据结构算法学习 之 红黑树
  16. 揪出占用电脑C盘的元凶“微信”,立马清爽了
  17. IOS积分墙:末落贵族与新兴势力PK
  18. 一男老师每日百词转载+连载(3)
  19. SSH框架电力项目八--运行监控的保存
  20. 如何使用N1盒子实现自动撸豆

热门文章

  1. Python——Window启动服务
  2. HTML5 -canvas拖拽、移动 绘制图片可操作移动,拖动
  3. Quartz和OpenGL绘图基本概念
  4. 获取字符串的真实长度
  5. 纵向合并gridview单元格的两种方法
  6. mysql 设置 0、1 用什么数据类型_什么是MySQL数据库?看这一篇干货文章就够了!...
  7. linux终端中出现 cd: OLDPWD 未设定 的提示
  8. 查找数组中重复数字 (二)
  9. Scheme调试手册(四)
  10. Abseil之string_view