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_())
其他文章: