1. 程式人生 > >解讀python中SocketServer源碼

解讀python中SocketServer源碼

cli eve sockets tro png onclick 計算機 第一個 類對象

在看SocketServer源碼之前,先看一個例子:

技術分享

技術分享
class Base(object):
    def __init__(self, name):
        self.name = name
        self.Testfunc()

    def Testfunc(self):
        print(do Base Testfunc)


class Son(Base):
    def Testfunc(self):
        print(do Son Testfunc)


sonobj = Son(sonobj
)
View Code

技術分享

技術分享

技術分享
class Base(object):
    def Testfunc(self):
        print(do Base Testfunc)


class Son(Base):
    def __init__(self, name):
        self.name = name
        self.Testfunc()

    def Testfunc(self):
        print(do Son Testfunc)

sonobj = Son(sonobj
)
View Code

這是一個簡單的類繼承,我們可以看到父類Base和子類Son,它們中各有一個Testfunc方法,當我們實例化子類的對象sonobj時,可以看到初始化方法中黃色框框調用了Testfunc,那麽這個時候執行的是哪個類裏面的代碼呢?我會告訴你執行的是子類裏面的方法!

其實這兩段代碼表示的是一個意思,盡管Son繼承了Base類,父子類中都有同樣的方法,但是由於我們實例化了子類的對象,所以這個在初始化方法裏的self.Testfunc,self指的是子類的對象,當然也就先調用子類中的方法啦。所以盡管在第一個例子中,初始化方法在父類執行,但是還是改變不了它是子類對象的本質,當我們使用self去調用Testfunc方法時,始終是先調用子類的方法。

看完剛剛的代碼,我們就知道了對象和self的真實意義,現在再來看看關於繼承的順序問題:

技術分享

技術分享
class Base(object):
    def Testfunc(self):
        print do Base Testfunc

class Son(Base):
    def __init__(self,name):
        self.name = name
        self.Testfunc()

    def Testfunc(self):
        print do Son Testfunc

class Base2(object):
    def Testfunc(self):
        print do Base2 Testfunc

class GrandSon(Base2,Son):
    pass

#sonobj = Son(‘sonobj‘)
sonobj = GrandSon(sonobj)
View Code

看上面的代碼,我們猜測一下,執行之後,控制臺會打印什麽呢?先揭曉答案,會打印Base2方法中的內容,原因很簡單:盡管這三個類中都有同樣的Testfunc方法,但是,由於計算機在找方法的時候,遵循的順序是:Base2,Son,Base,所以它會先找到Base2類,而這個類中剛好有它要找的方法,它也就拿去執行啦!

為了更好的來說明類之間的繼承和對象調用方法之間的聯系,我們需要畫個簡易的類圖:

技術分享

對照類圖看源碼

  根據上面的圖,我們就拿到了threadingTCPServer的相關類,並且搞清楚了它們之間的繼承關系和方法,接下來我們對照這張簡易類圖來看看代碼執行的過程:

  初始化相關過程:

技術分享

  執行serve_forever的相關代碼:

技術分享

就是這樣,我們一路按照調用軌跡去尋找,每次看到一個調用都先對照上面的簡易類圖,看看有沒有重名方法,如果有,就要找到最近的方法並查看裏面的內容,以此類推:按照這種方法,就會感覺所有代碼都在一個文件一樣

解讀python中SocketServer源碼