使用multiprocessing解决PyMuPDF不支持多线程加载导致的界面卡死无响应问题,及一个PyQt5实现的简易PDF阅读器例子...

最近在用PyMuPDF实现一个PDF阅读器,发现PyMuPDF在加载某些epub时耗时非常长,有的长达10几秒,会导致界面卡死无响应。

尝试用多线程后台加载,发现还是不能解决问题,和作者交流(issue链接 fitz.open blocks main thread even though I use it in a thread)后,作者说该库不支持真正的多线程,在多线程模式下也会阻塞主线程。

最后用multiprocessing解决该问题,我另外写了一个简单的PyQt5实现的PDF阅读器来说明如何解决该问题,效果图及代码如下。

#!python3
# -*- coding: utf-8 -*-
# this script demostrates how to use PyMuPDF in multiprocessing to avoid unresponsive GUI when fitz.open costs a long time
# author: yinkaisheng@live.com
import os
import sys
import time
import multiprocessing as mp
import queue
import fitz
from PyQt5 import QtCore, QtGui, QtWidgets

class DocForm(QtWidgets.QWidget):
    def __init__(self):
        super().__init__()
        self.process = None
        self.queNum = mp.Queue()
        self.queDoc = mp.Queue()
        self.pageCount = 0
        self.curPageNum = 0
        self.lastDir = ''
        self.timerSend = QtCore.QTimer(self)
        self.timerSend.timeout.connect(self.onTimerSendPageNum)
        self.timerGet = QtCore.QTimer(self)
        self.timerGet.timeout.connect(self.onTimerGetPage)
        self.timerWaiting = QtCore.QTimer(self)
        self.timerWaiting.timeout.connect(self.onTimerWaiting)
        self.initUI()

    def initUI(self):
        vbox = QtWidgets.QVBoxLayout()
        self.setLayout(vbox)

        hbox = QtWidgets.QHBoxLayout()
        self.btnOpen = QtWidgets.QPushButton('OpenDocument', self)
        self.btnOpen.clicked.connect(self.openDoc)
        hbox.addWidget(self.btnOpen)

        self.btnPlay = QtWidgets.QPushButton('PlayDocument', self)
        self.btnPlay.clicked.connect(self.playDoc)
        hbox.addWidget(self.btnPlay)

        self.btnStop = QtWidgets.QPushButton('Stop', self)
        self.btnStop.clicked.connect(self.stopPlay)
        hbox.addWidget(self.btnStop)

        self.label = QtWidgets.QLabel('0/0', self)
        self.label.setFont(QtGui.QFont('Verdana', 20))
        hbox.addWidget(self.label)

        vbox.addLayout(hbox)

        self.labelImg = QtWidgets.QLabel('Document', self)
        sizePolicy = QtWidgets.QSizePolicy(QtWidgets.QSizePolicy.Preferred, QtWidgets.QSizePolicy.Expanding)
        self.labelImg.setSizePolicy(sizePolicy)
        vbox.addWidget(self.labelImg)

        self.setGeometry(100, 100, 500, 600)
        self.setWindowTitle('PyMuPDF Document Player')
        self.show()

    def openDoc(self):
        path, _ = QtWidgets.QFileDialog.getOpenFileName(self, "Open Document", self.lastDir,
                    "All Supported Files (*.pdf;*.epub;*.xps;*.oxps;*.cbz;*.fb2);;PDF Files (*.pdf);;EPUB Files (*.epub);;XPS Files (*.xps);;OpenXPS Files (*.oxps);;CBZ Files (*.cbz);;FB2 Files (*.fb2)", options=QtWidgets.QFileDialog.Options())
        if path:
            self.lastDir, self.file = os.path.split(path)
            if self.process:
                self.queNum.put(-1) # use -1 to notify the process to exit
            self.timerSend.stop()
            self.curPageNum = 0
            self.pageCount = 0
            self.process = mp.Process(target=openDocInProcess, args=(path, self.queNum, self.queDoc))
            self.process.start()
            self.timerGet.start(40)
            self.label.setText('0/0')
            self.queNum.put(0)
            self.startTime = time.perf_counter()
            self.timerWaiting.start(40)

    def playDoc(self):
        self.timerSend.start(500)

    def stopPlay(self):
        self.timerSend.stop()

    def onTimerSendPageNum(self):
        if self.curPageNum < self.pageCount - 1:
            self.queNum.put(self.curPageNum + 1)
        else:
            self.timerSend.stop()

    def onTimerGetPage(self):
        try:
            ret = self.queDoc.get(False)
            if isinstance(ret, int):
                self.timerWaiting.stop()
                self.pageCount = ret
                self.label.setText('{}/{}'.format(self.curPageNum + 1, self.pageCount))
            else:#tuple, pixmap info
                num, samples, width, height, stride, alpha = ret
                self.curPageNum = num
                self.label.setText('{}/{}'.format(self.curPageNum + 1, self.pageCount))
                fmt = QtGui.QImage.Format_RGBA8888 if alpha else QtGui.QImage.Format_RGB888
                qimg = QtGui.QImage(samples, width, height, stride, fmt)
                self.labelImg.setPixmap(QtGui.QPixmap.fromImage(qimg))
        except queue.Empty as ex:
            pass

    def onTimerWaiting(self):
        self.labelImg.setText('Loading "{}", {:.2f}s'.format(self.file, time.perf_counter() - self.startTime))

    def closeEvent(self, event):
        self.queNum.put(-1)
        event.accept()

def openDocInProcess(path, queNum, quePageInfo):
    doc = fitz.open(path)
    quePageInfo.put(doc.pageCount)
    while True:
        num = queNum.get()
        if num < 0:
            break
        page = doc.loadPage(num)
        pix = page.getPixmap()
        quePageInfo.put((num, pix.samples, pix.width, pix.height, pix.stride, pix.alpha))
    doc.close()
    print('process exit')

if __name__ == '__main__':
    app = QtWidgets.QApplication(sys.argv)
    form = DocForm()
    sys.exit(app.exec_())

 

转载于:https://www.cnblogs.com/Yinkaisheng/p/10955863.html

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: PyQt5一个功能强大且易于使用Python GUI框架,但有时我们可能会遇到界面假死的情况。界面假死主要是因为长时间运行的任务导致主线程被阻塞,无法响应用户的操作。以下是一些解决界面假死的常见方法: 1. 使用多线程:将长时间运行的任务放到一个单独的线程中运行,以避免阻塞主线程。可以使用PyQt5中的QThread类来创建和管理线程。在长时间运行的任务完成后,通过信号槽机制来将结果传递回主线程更新界面。 2. 使用定时器:如果任务可以分割成多个小的操作,可以使用PyQt5中的QTimer类,将每个小操作放在一个定时器事件中处理。定时器将任务分解为一系列小操作,每个操作执行完后,释放主线程,使得界面能够及时响应用户。 3. 使用进程池:对于某些特殊情况,如大量IO操作或者计算密集型任务,可以使用PyQt5Python标准库中的multiprocessing模块来使用进程池。进程池可以将任务分配给多个子进程来执行,从而避免主线程被阻塞。 4. 优化代码:有时候界面假死是因为代码本身存在性能问题,需要对代码进行优化。可以使用工具来检测代码中的性能瓶颈,对关键部分进行改进,提高整体性能,从而降低假死的概率。 综上所述,使用多线程、定时器、进程池以及代码优化,可以有效解决PyQt5界面假死的问题,提升用户体验。在开发过程中,需要根据具体情况选择适合的方法来解决界面假死问题。 ### 回答2: PyQt5一个Python的GUI框架,当界面假死时,可能会导致用户无法响应操作,这对于GUI应用程序来说是非常严重的问题。下面是一些可能导致界面假死的原因以及解决方法: 1. 事件处理阻塞:如果在界面线程中执行了一个耗时操作,例如大量计算或者网络请求,会导致事件队列堆积,从而导致界面无法响应解决方法是将耗时操作放到单独的线程中进行,使用QThread或者QRunnable来开启一个新线程,并通过信号槽机制与界面进行通信。 2. UI更新频繁:如果界面需要频繁刷新,例如在循环中不断更新数据或者绘制图表,会导致界面假死。解决方法是使用定时器来控制刷新频率,避免界面过度更新。可以使用QTimer类来实现定时器功能。 3. 死循环:如果在代码中存在死循环,例如无限循环的条件或者递归调用,会导致界面假死。解决方法是检查代码逻辑,避免死循环的出现。 4. 插件或者扩展的冲突:如果在使用PyQt5的过程中使用了其他第三方库或者插件,可能会导致冲突,从而导致界面假死。解决方法是检查并更新第三方库的版本,或者采用兼容性更好的库。 总而言之,解决界面假死的关键是将阻塞操作移到单独线程,并合理控制UI的刷新频率。另外,检查代码中是否存在死循环或者第三方库冲突也是重要的。 ### 回答3: 在使用PyQt5开发界面时,有时会遇到界面假死的情况,即界面响应或无法操作。解决这个问题的方法有以下几点: 1. 使用多线程:在主线程中处理任务会导致界面假死,可以将耗时的操作放入子线程中运行,以保持界面响应性。可以使用Python的`threading`库或者Qt的`QThread`类来实现多线程。 2. 使用定时器:如果界面假死是由于某个循环或耗时操作导致的,可以使用定时器来定时检测并执行任务。PyQt5提供了`QTimer`类,可以设置定时器并在指定时间间隔内执行任务。 3. 优化界面布局和逻辑:界面假死也可能是由于界面布局复杂或者逻辑处理不当引起的。可以通过简化布局或者优化代码来提高界面响应性。例如,可以考虑将界面分成多个子部分,每个部分使用单独的线程处理。 4. 使用异步操作:对于一些需要等待结果的操作(例如网络请求),可以使用异步操作,以免阻塞主线程。Python的`asyncio`库和Qt的`QFuture`类都提供了异步操作的支持。 5. 避免过多的UI更新:界面假死有时也与过于频繁的UI更新有关。可以减少UI更新的次数,例如通过缓冲数据或者使用信号与槽来传递数据。 总之,解决界面假死的方法可以根据具体情况而定。以上是一些常见的解决方法,通过合理地选择和组合这些方法,可以有效地解决PyQt5界面假死问题
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值