簡體   English   中英

使用Python將變量從一個類實例傳遞給另一個實例?

[英]Passing variable from one class instance to another using Python?

我在將一個類實例中定義的變量傳遞給另一個類實例時遇到麻煩。 我對使用類比較陌生,但是據我了解,可以通過簡單地將其定義為類實例的一部分來將變量從一個實例傳遞到另一個實例(例如下面的示例)。 雖然我過去曾使用過此模型,但從未嘗試過使用諸如wxPython類的GUI框架來進行此操作。

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實例似乎是預定義的,不會接受任何其他參數(允許我僅將諸如title,size等之類的內容)傳遞給類實例。

我面臨的另一個問題是,我試圖通過調用對話框窗口將變量傳遞給三個類,然后從對話框中調用一個單獨的類來啟動工作線程。

所以我的問題是:

  1. 如何將變量從第一類實例傳遞到第三類實例?

  2. 如何覆蓋wxPython實例以允許定義其他變量?

  3. 或者,是否可以創建自定義事件處理程序以傳遞必要的數據?

澄清...

我正在使用Python,並且想以為我了解使用類和具有Tkinter和wxPython(在本項目中使用)之類的框架的GUI進行編程的基礎。 我編寫了一個主類/實例,該類從用戶那里獲取了一些數據,我希望能夠傳遞存儲在self.main_instance_var中的信息,並將其傳遞給第二個類/實例(在本例中為“進度對話框”窗口)從頭等艙叫來的。

當我嘗試在“進度對話框”中使用上述模型時,出現了一個非常無意義的語法錯誤(“在關鍵字arg之后為非關鍵字arg”)。 阻止我進一步將變量從“進度對話框”窗口傳遞到工作線程。 如果我有一個異常,那將是一回事,但是語法錯誤是我不理解的。 請看下面的簡短示例:

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()

就個人而言,我喜歡使用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/

在其wiki上也有一篇關於線程和wxPython的好文章: http : //wiki.wxpython.org/LongRunningTasks

問題中建議的方法是使用Python進行處理的首選方式。 諸如b.do_something(f.var)__init__(self, parent, title, myVar)類的語句是在類之間傳遞信息的絕佳方法。

剛開始時很常見,但是我的猜測是您在某處發生了一個小的語法錯誤,並認為這意味着您使用了錯誤的通用方法。 相反,您的一般方法看起來不錯,但是您只需要弄清楚是什么導致了特定錯誤。

對其他答案的評論:

1)Set / get函數也可以很好地工作,但是Python更喜歡使用屬性方法。 (就我個人而言,我仍然出於習慣使用set / get方法,但這不是Pythonic。)

2)pubsub很棒,但是它不是“在類之間傳遞信息”的通用工具, 例如 ,對於i = int(“ 2”),不想使用pubsub。 Pubsub更適用於其中有兩個需要傳遞一些信息的wxFrame的情況。 這是有原因的,它是wxPython而不是Python的一部分。

您可能已經在本月前完成了工作,但是我剛遇到了一個與wxPython對話框相同的問題,並且通過在函數中通知了全局使其工作:

elusive_variable="" # declare outside class & function

class MyForm(wx.Frame):

    def onOpenFile(self, event):
        global elusive_variable
        # ...other stuff here
        elusive_variable="Oh hi Mark"

if __name__ == "__main__":
app = wx.App(False)
frame = MyForm()
frame.Show()
app.MainLoop()
print elusive_variable # variable is liberated!!!

可能還有一些更開明的方法,但這可行...

這里是一個例子:

class One:
    param1 = ['a', 'b']

class Two:
    i = One()
    print i.param1

將以上代碼保存在.py文件中並運行。 您應該看到輸出,我想這可能是將變量從一個類交換到另一個類的簡單方法

在OOP中,如果要將數據傳入和傳出對象,則應在該類中定義一個set / get函數,以便可以從該對象獲取數據以及在該對象中設置數據。 因此,在您的情況下,對象的每個實例都將調用各自的get / set函數以在對象之間來回傳遞數據。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM