PYQT5学习文档3-多线程应用,防止主界面假死

python内核是没有多线程的应用,这里多线程的应用主要是用来处理程序运行时候,当代码运行时间过长的时候,会出现主界面的假死状态,这时候就需要用多线程来进行处理。

把之前的代码的time.sleep(0.1) 调整成为5秒之后,再次尝试运行,拖拽界面的时候可以发现,整个界面是卡顿的,会出现未响应的提示。

传送门:PYQT5学习文档2-界面组件使用说明-CSDN博客

多线程的使用,对之前的代码继续进行补充,将process里面的代码迁移到线程中运行

新建线程类,并建立好信号传输。

#建立一个线程
class My_thread(QThread):
    #建立一个信号,信号类型为int型,也可以是其他的型号类型,str,dic,list等。
    num_trig = pyqtSignal(int)

    def __init__(self):
        super(My_thread, self).__init__()

    def run(self) -> None:
        global n
        n=10
        for i in range(n):
            time.sleep(5)
            #传输信号 i+1 
            self.num_trig.emit(i+1)

在窗口类中,增加一个函数用来获取信号

    def setValue(self,v):
        #在textbrowser中显示内容
        self.textBrowser.append(f"输出:{v}")
        # 移动光标到底部
        self.textBrowser.moveCursor(self.textBrowser.textCursor().End) 

        #界面刷新显示(没有以下代码,界面会出现假死现象,循环运行完成之后才会全部显示出来,可以注释掉对比下)
        #多线程的内容等后面再做补充。
        QtWidgets.QApplication.processEvents()

        #设置进度条展示(进度条设置也有刷新的功能,设置进度条后,可以不用上面那个刷新代码,也会有刷新的效果)
        self.progressBar.setProperty("value", (v)/n*100)

在process中调用线程运行即可

    def process(self):
        
        #输出文件夹路径:
        self.textBrowser.append(f"输出:{self.lineEdit_folder.text()}")
        # 移动光标到底部
        self.textBrowser.moveCursor(self.textBrowser.textCursor().End) 
        
        #输出文件的路径:
        self.textBrowser.append(f"输出:{self.lineEdit_file.text()}")
        # 移动光标到底部
        self.textBrowser.moveCursor(self.textBrowser.textCursor().End) 
        
        # 实例化一个线程
        self.t = My_thread()    
        # 线程信号绑定到函数setValue上
        self.t.num_trig.connect(self.setValue)      
        #启动线程
        self.t.start()

 完整代码如下:

#读取包
from PyQt5 import QtWidgets

# 导入ui文件转换后的py文件
from test import Ui_Dialog

from PyQt5.QtWidgets import QFileDialog
from PyQt5 import QtWidgets, QtCore
from PyQt5.QtWidgets import QLabel,QHBoxLayout,QWidget,QApplication,QMainWindow,QMessageBox
from PyQt5.QtGui import QPixmap
from PyQt5.QtCore import Qt, QThread, pyqtSignal, QWaitCondition, QMutex

import time

import warnings
warnings.filterwarnings("ignore")


#创建窗口类
class mywindow(QtWidgets.QWidget, Ui_Dialog):
    quit_trig = pyqtSignal()
    
    def  __init__ (self):
        super(mywindow, self).__init__()
        #运行界面
        self.setupUi(self)
        
        #选择文件夹按钮与函数链接
        self.choise_folder.clicked.connect(self.write_folder)
        
        #选择文件按钮函数链接
        self.choise_file.clicked.connect(self.read_file)
        
        #开始按钮函数链接
        self.start.clicked.connect(self.process)
        
        #退出按钮函数链接
        self.exit.clicked.connect(self.clickButtonCloseWindow)
    
        #初始化进度条为0,进度默认范围0-100
        self.progressBar.setProperty("value", 0)
    
    
    def write_folder(self):
        #选取文件夹
        foldername = QFileDialog.getExistingDirectory(self, "选取文件夹", "C:/")
        #将选取的文件夹路径写入lineEdit_folder中
        self.lineEdit_folder.setText(foldername)
    
    def read_file(self):
        #选取文件路径
        filename, filetype =QFileDialog.getOpenFileName(self, "选取文件", "C:/", "All Files(*);;Text Files(*.csv)")
        #将路径写入 lineEdit_file 中
        self.lineEdit_file.setText(filename)
        
    def clickButtonCloseWindow(self):
        #调用qt的弹窗消息的库
        a = QMessageBox.question(self, '退出', '是否退出程序?', QMessageBox.Yes | QMessageBox.No, QMessageBox.No)      #"退出"代表的是弹出框的标题,"你确认退出.."表示弹出框的内容
        if a == QMessageBox.Yes:  
            self.close()        #接受关闭事件  
        else:  
            pass

    def process(self):
        
        #输出文件夹路径:
        self.textBrowser.append(f"输出:{self.lineEdit_folder.text()}")
        # 移动光标到底部
        self.textBrowser.moveCursor(self.textBrowser.textCursor().End) 
        
        #输出文件的路径:
        self.textBrowser.append(f"输出:{self.lineEdit_file.text()}")
        # 移动光标到底部
        self.textBrowser.moveCursor(self.textBrowser.textCursor().End) 
        
        # 实例化一个线程
        self.t = My_thread()    
        # 线程信号绑定到函数setValue上
        self.t.num_trig.connect(self.setValue)      
        #启动线程
        self.t.start()
        
    def setValue(self,v):
        #在textbrowser中显示内容
        self.textBrowser.append(f"输出:{v}")
        # 移动光标到底部
        self.textBrowser.moveCursor(self.textBrowser.textCursor().End) 

        #界面刷新显示(没有以下代码,界面会出现假死现象,循环运行完成之后才会全部显示出来,可以注释掉对比下)
        #多线程的内容等后面再做补充。
        QtWidgets.QApplication.processEvents()

        #设置进度条展示(进度条设置也有刷新的功能,设置进度条后,可以不用上面那个刷新代码,也会有刷新的效果)
        self.progressBar.setProperty("value", (v)/n*100)
    
    
#建立一个线程
class My_thread(QThread):
    #建立一个信号,信号类型为int型,也可以是其他的型号类型,str,dic,list等。
    num_trig = pyqtSignal(int)

    def __init__(self):
        super(My_thread, self).__init__()

    def run(self) -> None:
        global n
        n=10
        for i in range(n):
            time.sleep(5)
            #传输信号 i+1 
            self.num_trig.emit(i+1)

            
    
if __name__=="__main__":
    import sys
    app=QtWidgets.QApplication(sys.argv)
    ui = mywindow()    
    ui.show()
    sys.exit(app.exec_())

其他文章:

PYQT5学习文档1-安装编辑界面并生成py文件 

PYQT5学习文档2-界面组件使用说明-CSDN博客 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值