我编写了一个GUI程序,在后面做了很多计算,同时显示了一个进度条。完成后,新屏幕将显示结果。
然后我想在计算之前创建另一个界面,让用户选择是否要使用最后的计算结果,从而跳过计算。
我制作了一个屏幕,其中有一个连接到计算的按钮,以及一个用于选择最后一个计算结果文件的组合框。
然而,当我点击按钮时,它什么也没做。大约10秒后(计算的持续时间),结果屏幕会弹出。因此,它跳过了进度条屏幕。为什么呢?
这是原计划的一部分:
import sys
import configparser
import getpass
import telnetlib
import time
import subprocess
from datetime import *
from log_tracker import *
from PyQt4.QtCore import *
from PyQt4.QtGui import *
from PyQt4 import QtCore
from PyQt4 import QtGui
class Task_Checker(QMainWindow):
def __init__(self):
super(Task_Checker, self).__init__()
config = configparser.RawConfigParser()
config.read('profile.cfg')
self.log_path = config.get('Config', 'log_path')
self.log_prefix = config.get('Config', 'log_prefix')
self.log_suffix = config.get('Config', 'log_suffix')
self.initUI()
def check_production(self):
self.log_tracker = Log_Tracker(self)
self.log_tracker.tick.connect(self.pbar.setValue)
self.log_tracker.parseConfig()
self.log_tracker.connectDb()
self.log_tracker.trackLog()
def initUI(self):
self.resize(1400, 768)
self.center()
self.statusBar().showMessage('Checking Production Programs')
self.wait_message = QLabel('Checking Production Programs')
self.wait_message.setAlignment(Qt.Alignment(Qt.AlignHCenter))
self.pbar = QProgressBar(self)
self.pbar.setMinimum(0)
self.pbar.setMaximum(100)
vbox = QVBoxLayout()
vbox.addWidget(self.wait_message)
vbox.addWidget(self.pbar)
tmpWidget2 = QWidget()
tmpWidget2.setLayout(vbox)
self.setCentralWidget(tmpWidget2)
self.show()
self.check_production()
self.pbar.hide()
self.statusBar().showMessage('Processing Information')
self.tabs = QTabWidget()
mua_table = self.processInfo('MUA')
bps_table = self.processInfo('BPS')
obdua_table = self.processInfo('OBDUA')
sua_table = self.processInfo('SUA')
ngr_ftp_table = self.processInfo('NGR_FTP')
bpspdfbill_table = self.processInfo('BpsPdfBill')
disk_space_table = self.processInfo('Disk_Space')
self.tabs.addTab(mua_table, 'MUA')
self.tabs.addTab(bps_table, 'BPS')
self.tabs.addTab(obdua_table, 'ODBUA')
self.tabs.addTab(sua_table, 'SUA')
self.tabs.addTab(ngr_ftp_table, 'NGR_FTP')
self.tabs.addTab(bpspdfbill_table, 'BpsPdfBill')
self.tabs.addTab(disk_space_table, 'Disk_Space')
self.setCentralWidget(self.tabs)
self.statusBar().showMessage('Ready')
self.setWindowTitle('Task Checker')
self.show()
def center(self):
qr = self.frameGeometry()
cp = QDesktopWidget().availableGeometry().center()
qr.moveCenter(cp)
self.move(qr.topLeft())
def processInfo(self, project_name):
...the processing...
为了在进度条之前添加新屏幕(最初它立即加载计算),我对initUI()进行了一些更改并将计算部分移动到新的子例程checkProd(),然后连接它用一个按钮:
import sys
import configparser
import getpass
import telnetlib
import time
import subprocess
from datetime import *
from log_tracker import *
from PyQt4.QtCore import *
from PyQt4.QtGui import *
from PyQt4 import QtCore
from PyQt4 import QtGui
class Task_Checker(QMainWindow):
def __init__(self):
super(Task_Checker, self).__init__()
config = configparser.RawConfigParser()
config.read('profile.cfg')
self.log_path = config.get('Config', 'log_path')
self.log_prefix = config.get('Config', 'log_prefix')
self.log_suffix = config.get('Config', 'log_suffix')
self.initUI()
def check_production(self):
self.log_tracker = Log_Tracker(self)
self.log_tracker.tick.connect(self.pbar.setValue)
self.log_tracker.parseConfig()
self.log_tracker.connectDb()
self.log_tracker.trackLog()
def initUI(self):
self.resize(1400, 768)
self.center()
btn_check = QPushButton('Check Lastest Status', self)
btn_check.setToolTip('Click this if you want to check the lastest status in production')
combo = QComboBox()
dirlist = os.listdir(self.log_path)
for f in dirlist:
combo.addItem(f)
QtCore.QObject.connect(btn_check, QtCore.SIGNAL('clicked()'), self.checkProd)
hbox = QHBoxLayout()
hbox.addWidget(btn_check)
hbox.addWidget(combo)
tmpWidget = QWidget()
tmpWidget.setLayout(hbox)
self.setCentralWidget(tmpWidget)
self.show()
def checkProd(self):
self.statusBar().showMessage('Checking Production Programs')
self.wait_message = QLabel('Checking Production Programs')
self.wait_message.setAlignment(Qt.Alignment(Qt.AlignHCenter))
self.pbar = QProgressBar(self)
self.pbar.setMinimum(0)
self.pbar.setMaximum(100)
vbox = QVBoxLayout()
vbox.addWidget(self.wait_message)
vbox.addWidget(self.pbar)
tmpWidget2 = QWidget()
tmpWidget2.setLayout(vbox)
self.setCentralWidget(tmpWidget2)
self.show()
self.check_production()
self.pbar.hide()
self.statusBar().showMessage('Processing Information')
self.tabs = QTabWidget()
mua_table = self.processInfo('MUA')
bps_table = self.processInfo('BPS')
obdua_table = self.processInfo('OBDUA')
sua_table = self.processInfo('SUA')
ngr_ftp_table = self.processInfo('NGR_FTP')
bpspdfbill_table = self.processInfo('BpsPdfBill')
disk_space_table = self.processInfo('Disk_Space')
self.tabs.addTab(mua_table, 'MUA')
self.tabs.addTab(bps_table, 'BPS')
self.tabs.addTab(obdua_table, 'ODBUA')
self.tabs.addTab(sua_table, 'SUA')
self.tabs.addTab(ngr_ftp_table, 'NGR_FTP')
self.tabs.addTab(bpspdfbill_table, 'BpsPdfBill')
self.tabs.addTab(disk_space_table, 'Disk_Space')
self.setCentralWidget(self.tabs)
self.statusBar().showMessage('Ready')
self.setWindowTitle('Task Checker')
self.show()
def center(self):
qr = self.frameGeometry()
cp = QDesktopWidget().availableGeometry().center()
qr.moveCenter(cp)
self.move(qr.topLeft())
def processInfo(self, project_name):
...the processing...
答案 0 :(得分:0)
这是很多不必要的代码要阅读,但我设法解析的是,你在主线程中进行大量计算,同时期望主线程中的另一个小部件也能保持响应能力。 / p>
使用PyQt4(可能是大多数使用主线程事件循环的GUI框架)时,这是一个常见的陷阱。问题是,当您启动应用程序时,主事件循环会不断轮询从GUI处理的新事件。它期望每个操作都要花很短的时间才能完成,或者定期将控制权交还给eventloop。现在,您的计算正在占用主线程的所有可用性,并且您的进度窗口尝试执行的任何操作都只是备份到队列中,等待事件循环拾取它的机会。
对此的简单回答是:在单独的QThread中进行任何繁重的计算,并通过信号与主线程进行通信。这需要对你进行一些重组。你不应该把任何重量放在任何类的init中。但是你已经设法通过将计算附加到按钮来解决第二个例子。好的开始。该按钮实际上应该在另一个线程中启动calc,因此不会阻塞主线程。
GUI线程和工作线程
如上所述,每个程序都有一个线程 什么时候开始。这个线程被称为“主线程”(也是 在Qt应用程序中称为“GUI线程”)。 Qt GUI必须运行 这个帖子。例如,所有小部件和几个相关的类 QPixmap,不适用于辅助线程。辅助线程是 通常被称为“工人线程”,因为它习惯了 从主线程卸载处理工作 ...
使用主题
线程基本上有两种用例:
1.制作 通过使用多核处理器加快处理速度 2.保留GUI 线程或其他时间关键线程通过卸载响应很长 持久处理或阻止对其他线程的调用。
如果您只想查看某些结果,可以快速修复,您可以定期从计算方法中调用QtGui.QApplication.processEvents()
。这将经常允许事件循环清除待处理的操作。这样做会让你的进度小部件实际上起作用。基本上你正在做的是手动“抽”事件循环。这不是最好的方法。通常它保留给较轻的一次性东西。如果您想获得最佳性能,请将其移至线程。