2

我有一个 QWebEngine 类来阅读网页并为它们创建 BeautifulSoup。

这是代码:

import sys
from bs4 import BeautifulSoup
import os


from PyQt5 import QtCore, QtWidgets, QtWebEngineWidgets


class WebPage(QtWebEngineWidgets.QWebEnginePage):
    def __init__(self):
        super(WebPage, self).__init__()
        self.loadFinished.connect(self.handleLoadFinished)
        self.soup = []

    def start(self, urls):
        self._urls = iter(urls)
        self.fetchNext()

    def fetchNext(self):
        try:
            url = next(self._urls)
        except StopIteration:
            return False
        else:
            self.load(QtCore.QUrl(url))
        return True

    def processCurrentPage(self, html):
        url = self.url().toString()
        self.soup.append(BeautifulSoup(html, 'lxml'))
        if not self.fetchNext():
            QtWidgets.qApp.quit()

    def handleLoadFinished(self):
        self.toHtml(self.processCurrentPage)

这是另一个调用WebPage类的函数:

def get_soup(urls):
    app = QtWidgets.QApplication(sys.argv)
    webpage = WebPage()
    webpage.start(urls)
    return webpage.soup

这是main

if __name__ == "__main__":

    urls = ["http://www.hkexnews.hk/sdw/search/mutualmarket_c.aspx?t=sh", "http://www.hkexnews.hk/sdw/search/mutualmarket_c.aspx?t=sz"]      
    soups = get_soup(urls)

但是,当我执行程序时,程序会重新启动。

应该改变什么?

4

1 回答 1

2

这是我已经遇到的问题,分析我发现 QApplication 在 QWebEnginePage 使 QWebEngineProfile 被删除之前被破坏,在这种情况下导致 QWebEnginePage 崩溃。解决方案是通过将其设置为全局变量来使应用程序具有更大的范围。

另一方面,您必须调用 exec_() 以便允许信号操作的事件循环

# ...
app = None

def get_soup(urls):
    global app
    app = QtWidgets.QApplication(sys.argv)
    webpage = WebPage()
    webpage.start(urls)
    app.exec_()
    return webpage.soup
# ...

注意:与此问题相关的QTBUG-75547似乎已针对 Qt5>=5.12.4 解决,因此可能在PyQtWebEngine的下一个版本中将不再观察到该错误。

于 2019-06-17T09:02:17.583 回答