PyQt不会切换屏幕

时间:2012-07-06 04:26:46

标签: python-3.x pyqt4

我编写了一个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...

1 个答案:

答案 0 :(得分:0)

这是很多不必要的代码要阅读,但我设法解析的是,你在主线程中进行大量计算,同时期望主线程中的另一个小部件也能保持响应能力。 / p>

使用PyQt4(可能是大多数使用主线程事件循环的GUI框架)时,这是一个常见的陷阱。问题是,当您启动应用程序时,主事件循环会不断轮询从GUI处理的新事件。它期望每个操作都要花很短的时间才能完成,或者定期将控制权交还给eventloop。现在,您的计算正在占用主线程的所有可用性,并且您的进度窗口尝试执行的任何操作都只是备份到队列中,等待事件循环拾取它的机会。

对此的简单回答是:在单独的QThread中进行任何繁重的计算,并通过信号与主线程进行通信。这需要对你进行一些重组。你不应该把任何重量放在任何类的init中。但是你已经设法通过将计算附加到按钮来解决第二个例子。好的开始。该按钮实际上应该在另一个线程中启动calc,因此不会阻塞主线程。

Qt4.7 Threading Basics

  

GUI线程和工作线程
  如上所述,每个程序都有一个线程   什么时候开始。这个线程被称为“主线程”(也是   在Qt应用程序中称为“GUI线程”)。 Qt GUI必须运行   这个帖子。例如,所有小部件和几个相关的类   QPixmap,不适用于辅助线程。辅助线程是   通常被称为“工人线程”,因为它习惯了   从主线程卸载处理工作   ...
  使用主题
  线程基本上有两种用例:
  1.制作   通过使用多核处理器加快处理速度   2.保留GUI   线程或其他时间关键线程通过卸载响应很长   持久处理或阻止对其他线程的调用。

如果您只想查看某些结果,可以快速修复,您可以定期从计算方法中调用QtGui.QApplication.processEvents()。这将经常允许事件循环清除待处理的操作。这样做会让你的进度小部件实际上起作用。基本上你正在做的是手动“抽”事件循环。这不是最好的方法。通常它保留给较轻的一次性东西。如果您想获得最佳性能,请将其移至线程。