python - 使用 Qt 运行日志流送时 Python 崩溃
问题描述
目标
我有一个在运行时登录文件(realtime.log)的进程,我想在我的应用程序中实时打印该文件的每一行。换句话说,我想将进程的输出重定向到 GUI。这意味着我有两个不同的进程正在运行:“引擎”和 GUI。
我已经通过使用 Tkinter 实现了这一点,但由于我必须制作一个更复杂、更专业、更美观的 GUI,我决定切换到 Qt for Python ( PySide2 )。
问题
当我启动 GUI 并显示错误消息时,Python经常崩溃: Python has stop working。窗口开始打印线条,并在某些时候停止工作。
经过多次尝试和搜索,我发现程序只有在单击 GUI 窗口时才会崩溃。此外,程序不会突然崩溃,而是在引擎执行结束时崩溃。
环境
- 视窗 10
- Python 3.6.5
- PySide2 5.12.6
代码
请注意,这是一个简化版本。
datalog_path = "realtime.log"
def get_array_from_file(file_path):
try:
with open(file_path, 'r') as file:
lines = file.readlines()
return lines
except:
print('error in file access')
class Streamer(QRunnable):
def __init__(self, stream, old_array, edit):
super().__init__()
self.stream = stream
self.old_array = old_array
self.edit = edit
def run(self):
try:
while self.stream:
array_file = get_array_from_file(datalog_path)
if len(array_file) != len(self.old_array):
for line in array_file[len(self.old_array) - 1:len(array_file)]:
self.edit.append(line)
# print(line)
self.old_array.append(line)
except:
print('problem in streaming funct')
class Window(QMainWindow):
def __init__(self):
super().__init__()
layout = QVBoxLayout()
self.setWindowTitle("DATALOG")
self.thread_pool = QThreadPool()
self.edit = QTextEdit()
self.stream = True
self.old_array = get_array_from_file(datalog_path)
self.streamer = Streamer(self.stream, self.old_array, self.edit)
self.thread_pool.start(self.streamer)
window = QWidget()
layout.addWidget(self.edit)
window.setLayout(layout)
self.setCentralWidget(window)
def closeEvent(self, event):
self.stream = False
event.accept()
if __name__ == '__main__':
app = QApplication(sys.argv)
win = Window()
win.show()
app.exec_()
解决方案
@hyde 的答案指出了问题的原因,但它的解决方案不适用于 PySide2(在 PyQt5 中必须进行小修改,请参阅此),另一种方法是创建一个具有信号的 QObject:
class Signaller(QtCore.QObject):
textChanged = Signal(str)
class Streamer(QRunnable):
def __init__(self, stream, old_array):
super().__init__()
self.stream = stream
self.old_array = old_array
self.signaller = Signaller()
def run(self):
try:
while self.stream:
array_file = get_array_from_file(datalog_path)
if len(array_file) != len(self.old_array):
for line in array_file[len(self.old_array) - 1:len(array_file)]:
self.signaller.textChanged.emit(line)
# print(line)
self.old_array.append(line)
except:
print('problem in streaming funct')
self.stream = True
self.old_array = get_array_from_file(datalog_path)
self.streamer = Streamer(self.stream, self.old_array)
self.streamer.signaller.textChanged.connect(self.edit.append)
self.thread_pool.start(self.streamer)
推荐阅读
- javascript - 我应该将反应钩子作为值传递给上下文提供者吗?
- java - 使用 2D 点查找段的长度时出现 NullPointerException
- python - Python,执行失败(检测到致命错误)
- c++ - 我将如何遍历我的数组(缓冲区——包含一个文本文件),并以 30 个字节的块打印出来?
- html - 重定向回具有验证错误的特定 div 部分
- javascript - 选择多个按钮
- php - 如何在 Woocommerce 中按属性和供应商地址进行搜索
- php - Laravel 在集合序列化中保留组键
- sql - dd-mm-yyyy 到 yyyy-mm-dd SQL
- git - git move unpushed commited changes to a remote branch,在 master 上工作,然后返回分支