Python,修饰函数中的引用类实例/方法

2024-09-28 22:24:37 发布

您现在位置:Python中文网/ 问答频道 /正文

我很难找到一种方法来引用decorator函数中的类实例。你知道吗

import json
import time
import sys
from PyQt5.QtCore import *
from PyQt5.QtWidgets import *
from main_UI import Ui_ApplicationWindow
from slack import RTMClient

class WorkerThread(QThread):
  finished = pyqtSignal(str)   

  def __init__(self):
    QThread.__init__(self)  
    self.rtm_client = RTMClient(token="xoxp...") 

  def run(self):     
    self.rtm_client.start()

  @RTMClient.run_on(event="message")
  def say_hello(**payload):
    data = payload['data']
    if (len(data) != 0):
      if "text" in data:          
        text = data['text']         
        self.finished.emit(str(text))  


class ApplicationWindow(QMainWindow):

  def __init__(self):
    super(ApplicationWindow, self).__init__()
    self.ui = Ui_ApplicationWindow()
    self.ui.setupUi(self)   
    self.ui.pushButton.clicked.connect(self.start_rtm)      

  def start_rtm(self):
    self.thread = WorkerThread()
    self.thread.finished.connect(self.update)
    self.thread.start()

  @pyqtSlot(str)
  def update(self, data):
    self.ui.label.setText(data)

if __name__ == "__main__":
    app = QApplication(sys.argv)
    myWindow = ApplicationWindow()
    myWindow.show()
    app.exec_()

所以在say_hello中,因为它不能将self作为参数,所以我不能在函数末尾使用self.finished.emit(text)。你知道吗

如何使用say_hello中的self引用类实例/函数?你知道吗


Tags: 函数textfromimportselfuidatainit
2条回答

不,你不能。与其使用^{}装饰器,不如使用^{}函数来注册它。你知道吗

import threading
import asyncio

from slack import RTMClient

from PyQt5 import QtCore, QtWidgets


class SlackClient(QtCore.QObject):
    textChanged = QtCore.pyqtSignal(str)

    def start(self):
        RTMClient.on(event="message", callback=self.say_hello)
        threading.Thread(target=self._start_loop, daemon=True).start()

    def _start_loop(self):
        loop = asyncio.new_event_loop()
        asyncio.set_event_loop(loop)
        slack_token = "xoxb-...."
        rtm_client = RTMClient(token=slack_token)
        rtm_client.start()

    def say_hello(self, **payload):
        data = payload["data"]
        if data:
            if "text" in data:
                text = data["text"]
                self.textChanged.emit(text)


if __name__ == "__main__":
    import sys

    app = QtWidgets.QApplication(sys.argv)

    client = SlackClient()

    button = QtWidgets.QPushButton("Start")
    textedit = QtWidgets.QPlainTextEdit()

    button.clicked.connect(client.start)
    client.textChanged.connect(textedit.appendPlainText)

    w = QtWidgets.QWidget()
    lay = QtWidgets.QVBoxLayout(w)
    lay.addWidget(button)
    lay.addWidget(textedit)
    w.show()

    sys.exit(app.exec_())

更新:

import sys
import threading
import asyncio

from slack import RTMClient

from PyQt5 import QtCore, QtWidgets
from main_UI import Ui_ApplicationWindow


class SlackClient(QtCore.QObject):
    textChanged = QtCore.pyqtSignal(str)

    def start(self):
        RTMClient.on(event="message", callback=self.say_hello)
        threading.Thread(target=self._start_loop, daemon=True).start()

    def _start_loop(self):
        loop = asyncio.new_event_loop()
        asyncio.set_event_loop(loop)
        slack_token = "xoxb-...."
        rtm_client = RTMClient(token=slack_token)
        rtm_client.start()

    def say_hello(self, **payload):
        data = payload["data"]
        if data:
            if "text" in data:
                text = data["text"]
                self.textChanged.emit(text)


class ApplicationWindow(QtWidgets.QMainWindow):
    def __init__(self):
        super(ApplicationWindow, self).__init__()
        self.ui = Ui_ApplicationWindow()
        self.ui.setupUi(self)

        self.client = SlackClient()
        # connections
        self.ui.pushButton.clicked.connect(self.client.start)
        self.client.textChanged.connect(self.ui.label.setText)


if __name__ == "__main__":
    app = QtWidgets.QApplication(sys.argv)
    myWindow = ApplicationWindow()
    myWindow.show()
    sys.exit(app.exec_())

实际上你不能自我,因为那是一个全局变量,而不是类变量。你知道吗

from slack import RTMClient

class WorkerThread(QThread):
    finished = pyqtSignal(dict)

    def __init__(self):
        QThread.__init__(self)
        self.rtm_client = RTMClient(token="xoxp-....")    

    def run(self):
        self.rtm_client.start()

    @RTMClient.run_on(event="message")
    def say_hello(**payload):
        data = payload['data']
        if (len(data) != 0):
            if "text" in data:          
                text = data['text']                      
                WorkerThread.finished.emit(text)  < - using self impossible 

我建议您通过在开头添加两个下划线(__my_private_var)来将此类变量设为私有变量

相关问题 更多 >