Python 3.6.2 - 多个 __init__;一次多班
Posted
技术标签:
【中文标题】Python 3.6.2 - 多个 __init__;一次多班【英文标题】:Python 3.6.2 - multiple __init__; multiple classes at once 【发布时间】:2017-08-03 10:03:50 【问题描述】:我有两个分开运行的脚本。一个是 PyQt5 GUI 应用程序的代码,第二个是与this one 非常相似的代码,稍作修改以便能够转换内容,以防出现任何引起问题的笑脸。
基本上,当我在应用程序窗口中按下某个按钮时,我希望第二个代码会运行。
无论我如何努力适应第二个代码,它总是会导致我的应用程序(或 Python)崩溃。我能达到的最远距离是,当我关闭主窗口后第二个代码工作时 - 然后它运行,并给我想要的结果。
我怀疑这与来自第二个代码的__init__
不高兴主窗口中已经有另一个__init__
正在运行有关?
正如你所看到的,我对 Python 的面向对象部分非常困惑,尽管过去几天我在这个主题上努力自学,但我无法将这两个代码组合在一起。
我的应用:
#'all the necessary imports'
class MainWindow(QWidget):
def __init__(self, parent=None):
super(MainWindow, self).__init__(parent)
self.text = QWebEngineView(self)
self.proc_btn = QPushButton('Proceed')
self.userUrl = QLineEdit(self)
self.labOne = QLabel(self)
self.labTwo = QLabel(self)
self.defUrl = 'default'
self.init_ui()
def init_ui(self):
v_layout = QVBoxLayout()
h_layout = QHBoxLayout()
h_layout.addWidget(self.proc_btn)
h_layout.addWidget(self.userUrl)
v_layout.addWidget(self.text)
v_layout.addWidget(self.labOne)
v_layout.addWidget(self.labTwo)
v_layout.addLayout(h_layout)
self.labOne.setText('URL: ')
self.labTwo.setText('<ENTER LINK PLEASE>')
self.userUrl.returnPressed.connect(self.linkPut)
self.proc_btn.clicked.connect(self.doStuff)
self.setLayout(v_layout)
self.setWindowTitle('Scrapper')
self.show()
def doStuff(self):
print('Doing stuff (expecting 2nd script to be ran)')
def linkPut(self):
newText = (self.userUrl.text())
print('newText: ' + newText)
self.labTwo.setText(newText)
self.defUrl = newText
app = QApplication(sys.argv)
a_window = MainWindow()
sys.exit(app.exec_())
我需要实现的脚本:
#'all necessary imports'
class Page(QWebEnginePage):
def __init__(self, url):
self.app = QApplication(sys.argv)
QWebEnginePage.__init__(self)
self.html = ''
self.loadFinished.connect(self._on_load_finished)
self.load(QUrl(url))
self.app.exec_()
print('__init__ WORKS')
def _on_load_finished(self):
self.html = self.toHtml(self.Callable)
print('Load finished')
def Callable(self, html_str):
self.html = html_str
self.app.quit()
_nonbmp = re.compile(r'[\U00010000-\U0010FFFF]')
def _surrogatepair(match):
char = match.group()
assert ord(char) > 0xffff
encoded = char.encode('utf-16-le')
return (
chr(int.from_bytes(encoded[:2], 'little')) +
chr(int.from_bytes(encoded[2:], 'little')))
def with_surrogates(text):
return _nonbmp.sub(_surrogatepair, text)
def main():
page = Page('https://somenicepage.com/')
soup = bs.BeautifulSoup(page.html, 'html.parser'))
longStrCoded = str(soup.find("img", "class":"pictures"))
longStr = with_surrogates(longStrCoded)
print('long str: ' + longStr)
extract = longStr.split('src="')[1].split('"')[0]
print(extract)
if __name__ == '__main__': main()
【问题讨论】:
import other_py_file
在文件的开头,然后other_py_file.main()
要触发它
这个我已经试过了,如果我在我的应用程序中使用它会崩溃,但是如果我先关闭我的应用程序,然后运行other_py_file.main()
它可以工作。
你能比“它崩溃”更具体吗
''Python 已停止工作 - 程序停止正常工作导致的问题。如果有可用的解决方案,Windows 将关闭程序并通知您。'' 如果我有时在语法上打错字等,也会发生这种情况。由于某种原因在使用 PyQt5 时发生。
那我会尽量简化这个,一步一步增加复杂度,看看是哪一步导致了问题
【参考方案1】:
问题在于,在组合这两个文件时,您试图创建QApplication
的多个实例,这是不允许的。此外,QApplication
类旨在封装您的整个应用程序逻辑并负责事件处理等。您通常不应像在 Page.__init__
中那样在其他类中创建。
通常,您会在程序入口点附近创建并启动QApplication
。您在第一个代码块中正确地执行了此操作。
if __name__ == '__main__':
app = QtWidgets.QApplication(sys.argv) # Instantiate application
window = MainWindow() # The rest of your program logic should flow from here
sys.exit(app.exec_()) # Start application event loop
QtWebEngine
的异步特性使事情有点复杂,因为您的程序不会等待页面加载,然后再继续执行下一条指令。我相信人们在页面类中启动QApplication
是一种快速而肮脏(或幼稚)的方式来强制程序等待页面完成加载。这在一个 python 脚本中可能很好,其中Qt
仅用于QtWebEngine
评估动态网页的能力,但对于真正的Qt
应用程序来说是不好的做法。处理此问题的正确方法是通过回调或Qt
的信号和插槽系统。
基于您的原始类,这是一个使用回调在完全加载后继续处理 html 的版本。
class Page(QtWebEngineWidgets.QWebEnginePage):
def __init__(self, url):
super(Page, self).__init__()
self.url = QtCore.QUrl(url)
self.callback = None
self.html = ''
self.loadFinished.connect(self.on_load_finished)
def load_html(self, callback=None):
self.callback = callback
self.load(self.url)
def on_load_finished(self):
self.toHtml(self.on_html_ready)
def on_html_ready(self, html):
self.html = html
if self.callback:
self.callback(html)
接下来,定义将处理加载页面的回调。您可以在此处放置 main()
函数中的代码。
def do_stuff(html):
soup = bs.BeautifulSoup(html, 'html.parser'))
longStrCoded = str(soup.find("img", "class":"pictures"))
longStr = with_surrogates(longStrCoded)
print('long str: ' + longStr)
extract = longStr.split('src="')[1].split('"')[0]
print(extract)
最后,您将在MainWindow
类中加载这样的页面。
def doStuff(self):
self.page = Page(self.userUrl.text())
self.page.load_html(callback=do_stuff)
注意这里使用self
。如果我们不将页面实例存储在类中,它将在加载完成之前被删除,并且永远不会调用回调。
【讨论】:
谢谢,不仅现在可以用了,还帮我理解了很多。以上是关于Python 3.6.2 - 多个 __init__;一次多班的主要内容,如果未能解决你的问题,请参考以下文章