解讀python中SocketServer源碼
在看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(‘sonobjView 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源碼