运行bat文件的程序,其中包含使用Qthread运行可执行文件(longTask)的指令,但是当我使用Pyinstaller使用以下命令创建可执行文件时,它无法正常工作。我给了'--windowed'不为标准i / o提供控制台窗口
pyinstaller --onefile --windowed main.py
但有趣的是,当我删除--windowed参数
时,它按预期工作pyinstaller --onefile main.py
以下是代码:
from PyQt4.Qt import *
import subprocess
def callSubprocess():
page = QWizardPage()
page.setTitle("Run myLongTask")
runButton = QPushButton("Run")
progressBar = QProgressBar()
procLabel = QLabel()
procLabel1 = QLabel()
progressBar.setRange(0, 1)
layout = QGridLayout()
layout.addWidget(runButton, 0, 0)
layout.addWidget(progressBar, 0, 1)
layout.addWidget(procLabel)
layout.addWidget(procLabel1)
# Calls thread class
myLongTask = TaskThread()
runButton.clicked.connect(lambda: OnStart(myLongTask, progressBar, procLabel1))
myLongTask.taskFinished.connect(lambda: onFinished(progressBar, procLabel))
page.setLayout(layout)
return page
def OnStart(myLongTask, progressBar, procLabel1):
progressBar.setRange(0, 0)
myLongTask.start()
# I am waiting until my subprocess completes
while not myLongTask.isFinished():
QCoreApplication.processEvents()
procLabel1.setText("Hello This is main")
def onFinished(progressBar, procLabel):
# Stop the pulsation
progressBar.setRange(0, 1)
procLabel.setText("longTask finished")
class TaskThread(QThread):
taskFinished = pyqtSignal()
def __init__(self):
QThread.__init__(self)
def run(self):
proc = subprocess.Popen(r'C:\Users\Desktop\runInf.bat', bufsize=0, shell=True, stdout=subprocess.PIPE)
proc.wait()
self.taskFinished.emit()
def __del__(self):
self.wait()
if __name__ == '__main__':
import sys
app = QApplication(sys.argv)
wizard = QWizard()
wizard.addPage(callSubprocess())
wizard.setWindowTitle("Example Application")
wizard.show()
sys.exit(wizard.exec_())
在Pycharm中执行上面的代码时,它按预期工作。但是当使用PyInstaller构建时,主线程不会等到子进程完成。
任何想法如何创建一个可执行文件,其中线程按预期工作。 提前致谢
答案 0 :(得分:2)
我刚刚解决了这个问题。我的发现如下。
罪魁祸首是子流程。所以我用一些文件处理替换了子进程(让我们说打开一个文件10000次并写一些内容)它按预期工作,这意味着主线程等到子进程完成。
所以 Pyinstaller可以正常使用线程
我添加了更多参数
proc = subprocess.Popen(os.getcwd() + r'\runInf.bat',
shell=True,
stdout=subprocess.PIPE,
stderr=subprocess.STDOUT,
stdin=subprocess.PIPE,
cwd=os.getcwd(),
env=os.environ)
proc.stdin.close()
此更改使子进程运行,主线程也等待直到完成
答案 1 :(得分:0)
我遇到了类似的问题,我通过在 .spec 文件中使用“console=True”语句解决了这个问题。没有控制台,线程不会执行。