如何在Python中将变量从一个类实例传递到另一个类实例?

1 投票
5 回答
10524 浏览
提问于 2025-04-17 10:18

我在把一个类实例中定义的变量传递给另一个类实例时遇到了麻烦。我对使用类还比较陌生,但我知道变量可以通过简单地在类实例中定义来传递(就像下面的例子)。虽然我以前用过这种方式,但我从来没有尝试过在像 wxPython 这样的图形用户界面框架中使用。

class Foo(object):
    def __init__(self, var):
        self.var = var

class Bar(object):
    def do_something(self, var):
        print var*3

if __name__ == '__main__':
    f = Foo(3)
    b = Bar()
    b.do_something(f.var)

我遇到的问题是,wxPython 的实例似乎是预先定义好的,只能接受一些基本参数(比如标题、大小等),而不能接受其他额外的参数。

我面临的另一个问题是,我试图通过调用一个对话窗口来传递变量,而这个对话窗口又调用了一个专门用于启动工作线程的单独类。

所以我想问的问题是:

  1. 我该如何把第一个类实例中的变量传递给第三个类实例?

  2. 我该如何重写 wxPython 的实例,以便可以定义额外的变量?

  3. 或者,我能否创建一个自定义事件处理器来传递必要的数据?

为了更清楚...

我在使用 Python,并且希望我能理解使用类和图形用户界面框架(比如 Tkinter 和 wxPython,这个项目中用的是 wxPython)的编程基础。我写了一个主类/实例,从用户那里获取一些数据,我希望能够把存储在 self.main_instance_var 中的信息传递给第二个类/实例(在这个情况下是从第一个类调用的进度对话窗口)。

当我尝试在我的进度对话框中使用上述模型时,出现了一个非常不明确的语法错误('非关键字参数在关键字参数之后'),这让我无法将变量从进度对话框继续传递给工作线程。如果我遇到的是异常,那还好,但这个语法错误我就搞不懂了。下面是一个简短的例子:

class ProgressDialog(wx.Dialog):

    def __init__(self, parent, title, myVar):     # Generates syntax error on this line
        super(ProgressDialog, self).__init__(parent=parent, 
            title=title, size=(500, 110))
        self.var = myVar

基本源代码(应要求提供,抱歉代码有点乱):

import time
import os, sys, wx
from ftplib import FTP_TLS

from threading import Thread
from wx.lib.pubsub import Publisher

########################################################################
class FtpValues(object):
    """ Returns a property attribute - called by FtpFileTransfer
    Used to set values/variables for Host, USERID, PASSWD, FILE """

    #----------------------------------------------------------------------
    def __init__(self):
        self.varList = None

    #----------------------------------------------------------------------    
    def GetValues(self):
        return self.varList

    #----------------------------------------------------------------------
    def SetValues(self, HOST, USERID, PASSWD, FILE):
        self.varList = [HOST, USERID, PASSWD, FILE]

    #----------------------------------------------------------------------
    def DelValues(self):
        del self.valList

    Values = property(GetValues, SetValues, DelValues, "Set/Get FtpValues")

    # http://docs.python.org/library/functions.html#property

########################################################################
class FtpFileTransfer(Thread):
    """Test Worker Thread Class."""

    #----------------------------------------------------------------------
    def __init__(self):
        """Init Worker Thread Class."""
        Thread.__init__(self)
        self.StartTransfer()        # start the thread

    #----------------------------------------------------------------------
    def StartTransfer(self):        # was named run - started automatically                                    
        """Run Worker Thread."""    # when called by the start method
        # This is the code executing in the new thread.

        HOST, USERID, PASSWD, FILE = FtpValues.Values
        BLOCKSIZE = 57344
        try:
            ftp = FTP_TLS(HOST)
            ftp.login(USERID, PASSWD)
            ftp.prot_p()
            ftp.voidcmd("TYPE I")
            f = open(FILE, 'rb')
            datasock, esize = ftp.ntransfercmd(
                    'STOR %s' % os.path.basename(FILE))
            size = os.stat(FILE)[6]
            bytes_so_far = 0
            while 1:
                buf = f.read(BLOCKSIZE)
                if not buf:
                    break
                datasock.sendall(buf)
                bytes_so_far += len(buf)
                msg = [bytes_so_far, size]
                Publisher().sendMessage("update", msg)
        except: raise
        finally:
            try:
                datasock.close()
                f.close()
                ftp.voidresp()
                ftp.quit()
                print 'Complete...'
            except: pass

        wx.CallAfter(Publisher().sendMessage, "update", "Database Transfer Complete!")


########################################################################
class ProgressDialog(wx.Dialog):

    def __init__(self, parent, title):
        super(ProgressDialog, self).__init__(parent=parent, 
            title=title, size=(500, 110))

        self.displayLbl = wx.StaticText(self, -1, 'Verifying Database Files... ', (20, 20)) #Preparing for Transfer...
        self.gauge = wx.Gauge(self, -1, 100, (20, 45), (370, 24))        
        self.btn = btn = wx.Button(self, -1, 'Cancel', (400, 45), (-1, 25))
        btn.Bind(wx.EVT_BUTTON, self.OnClose)

        # listens for response from worker thread
        Publisher().subscribe(self.updateDisplay, "update")

        FtpFileTransfer()#.StartTransfer(HOST, USERID, PASSWD, FILE)        #Start the FTP Worker Thread
        #self.OnStart()

    #----------------------------------------------------------------------
    def run(self):
        FtpFileTransfer(HOST, USERID, PASSWD, FILE)

    #----------------------------------------------------------------------
    def OnClose(self, event):
        """ Place Holder """
        if self.btn.GetLabel() == 'Finish':
            # Do Something!
            pass
        return None

    #----------------------------------------------------------------------
    def updateDisplay(self, msg):
        """ Receives data from thread and updates the display """
        if isinstance(msg.data, list):
            bytes_so_far, size = msg.data
            k = 100 * bytes_so_far / size
            self.displayLbl.SetLabel("Sent %d of %d bytes %.1f%%" % (bytes_so_far, size, 100 * bytes_so_far / size))
            self.gauge.SetValue(k)
        else:
            self.displayLbl.SetLabel("%s" % msg.data)
            #self.btn.Enable()
            self.btn.SetLabel('Finish')


########################################################################
class MyForm(wx.Frame):

    #----------------------------------------------------------------------
    def __init__(self):
        wx.Frame.__init__(self, None, wx.ID_ANY, "Tutorial")

        # Add a panel so it looks the correct on all platforms
        panel = wx.Panel(self, wx.ID_ANY)
        self.displayLbl = wx.StaticText(panel, label="Amount of time since thread started goes here")
        self.btn = btn = wx.Button(panel, label="Start Thread")
        self.gauge = wx.Gauge(panel, -1, 100, size=(370, 24))

        btn.Bind(wx.EVT_BUTTON, self.onButton)

        sizer = wx.BoxSizer(wx.VERTICAL)
        sizer.Add(self.displayLbl, 0, wx.ALL|wx.CENTER, 5)
        sizer.Add(btn, 0, wx.ALL|wx.CENTER, 5)
        sizer.Add(self.gauge, 0, wx.ALL|wx.CENTER, 5)
        panel.SetSizer(sizer)

        self.VarData()
        # create a pubsub receiver
        Publisher().subscribe(self.updateDisplay, "update")

    #----------------------------------------------------------------------
    def onButton(self, event):
        """
        Runs the thread
        """

        chgdep = ProgressDialog(None, title='File Transfer. . .')
        chgdep.ShowModal()
        #chgdep.Destroy()

    #----------------------------------------------------------------------
    def updateDisplay(self, msg):
        """
        Receives data from thread and updates the display
        """
        if isinstance(msg.data, list):
            bytes_so_far, size = msg.data
            k = 100 * bytes_so_far / size
            self.displayLbl.SetLabel("Sent %d of %d bytes %.1f%%" % (bytes_so_far, size, 100 * bytes_so_far / size))
            self.gauge.SetValue(k)
        else:
            self.displayLbl.SetLabel("%s" % msg.data)
            self.btn.Enable()

    #----------------------------------------------------------------------
    def VarData(self):
        HOST = '127.0.0.1'
        USERID = 'test'
        PASSWD = 'P@ssw0rd'
        FILE = r'F:\Programming\temp\Test.zip'
        varList = [HOST, USERID, PASSWD, FILE]
        FtpValues.Values = HOST, USERID, PASSWD, FILE

#----------------------------------------------------------------------
# Run the program
if __name__ == "__main__":
    app = wx.PySimpleApp()
    frame = MyForm().Show()
    app.MainLoop()

5 个回答

0

在面向对象编程中,如果你想把数据传进对象里或者从对象里取出来,你需要在这个类里定义一些获取和设置数据的函数。这样你就可以从这个对象中获取数据,也可以把数据放进这个对象里。举个例子,你的每个对象实例都会调用相应的获取和设置函数,来在对象之间传递数据。

1

问题中提到的方法是用Python编程时的推荐做法。像 b.do_something(f.var)__init__(self, parent, title, myVar) 这样的写法是 完全可以的,用来在不同的类之间传递信息。

刚开始学习的时候,这种情况很常见。我猜你可能在某个地方犯了一个小的语法错误,结果让你觉得自己整体的方法不对。其实,你的方法看起来没问题,只需要找出导致具体错误的原因就行了。

对其他回答的评论:

1) 设置/获取函数也很好用,但Python更喜欢使用属性的方式。(个人来说,我还是习惯用设置/获取方法,但这并不是最符合Python风格的做法。)

2) pubsub很不错,但它并不是一个通用的“在类之间传递信息”的工具,比如说,你不会用pubsub来处理像 i = int("2") 这样的事情。pubsub更适合用在需要让两个wxFrame之间传递一些信息的场景。它之所以是wxPython的一部分,而不是Python的原因就在这里。

2

我个人喜欢用 wx.lib.pubsub 来在不同的类之间传递信息。我在我的应用程序中经常这么做。你可以在这里了解更多:http://www.blog.pythonlibrary.org/2010/06/27/wxpython-and-pubsub-a-simple-tutorial/

如果你需要从一个线程中发送数据,你就需要使用一些线程安全的方法,比如 wx.CallAfter、wx.CallLater 或 wx.PostEvent。你可以把这些方法和 pubsub 结合起来,在这些线程安全的方法里面调用你的 pubsub 发布者。我在这里展示了怎么做:http://www.blog.pythonlibrary.org/2010/05/22/wxpython-and-threads/

他们的维基上还有一篇关于线程和 wxPython 的好文章:http://wiki.wxpython.org/LongRunningTasks

撰写回答