盡管Python在Function Programming中有着其他語言難以企及的的優勢,但是我們也不要忘了Python也是一門OO語言哦。因此我們關注Python在FP上的優勢的同時,還得了解一下Python在OO方面的特性。
要討論Python的OO特性,了解Python中的Class自然是首當其沖了。在Python中定義class和創建對象實例都很簡單,具體代碼如下:
1 class GrandPa: 2 def __init__(self): 3 print('I\'m GrandPa') 4 5 6 class Father(GrandPa): 7 def __init__(self): 8 print('I\'m Father!') 9 10 class Son(Father): 11 """A simple example class""" 12 i = 12345 13 def __init__(self): 14 print('這是構造函數,son') 15 def sayHello(self): 16 return 'hello world' 17 18 if __name__ == '__main__': 19 son = Son() 20 # 類型幫助信息 21 print('類型幫助信息: ',Son.__doc__) 22 #類型名稱 23 print('類型名稱:',Son.__name__) 24 #類型所繼承的基類 25 print('類型所繼承的基類:',Son.__bases__) 26 #類型字典 27 print('類型字典:',Son.__dict__) 28 #類型所在模塊 29 print('類型所在模塊:',Son.__module__) 30 #實例類型 31 print('實例類型:',Son().__class__)
運行情況:
Python 3.3.2 (v3.3.2:d047928ae3f6, May 16 2013, 00:03:43) [MSC v.1600 32 bit (Intel)] on win32 Type "copyright", "credits" or "license()" for more information. >>> ================================ RESTART ================================ >>> 這是構造函數,son 類型幫助信息: A simple example class 類型名稱: Son 類型所繼承的基類: (<class '__main__.Father'>,) 類型字典: {'__module__': '__main__', 'sayHello': <function Son.sayHello at 0x010194F8>, '__doc__': 'A simple example class', '__init__': <function Son.__init__ at 0x010194B0>, 'i': 12345} 類型所在模塊: __main__ 這是構造函數,son 實例類型: <class '__main__.Son'> >>>
#Python支持多重繼承
首先第一點,你會發現Class的定義中有一個括號,這是體現繼承的地方。 Java用extends,C#、C++用冒號(:),Python則用括號了。從括號中包含着兩個值,聰明的你一定可以發現:Python支持多重繼承;
#__init__是Class中的構造函數
第二點,__init__是Class中的構造函數,兩種不同形式的構造函數體現了Python支持函數重載。在構造函數中,有一個特別的參數self,其含義與我們在Java和C#中常見的this是一樣的。在這里需要強調一點:在Class中定義的方法實質上也是function,但是在方法定義的時候必須包含self這個參數,而且必須將self這個參數放在第一位;
#python成員變量
第三點,在Python中,你並不需要顯式的聲明Class的Data Members,而是在賦值的時候,被賦值的變量就相應成為了Class的Data Memebers,正如代碼中的x和y。不僅你不需要顯式的聲明Data Members,更加特別的,你甚至可以通過del方法將Class中的Data Memebers給刪掉。當我第一次看到這樣的特性的時候,着實吃了一驚。畢竟OO的第一條就是封裝了,但是這樣的特性是不是破壞了封裝的特性呢?
#python方法二義性問題
第四點,由於Python支持多重繼承,因此就有可能出現方法二義性問題[1]。然而由於Python遵循深度優先的搜尋法則,很好地避免了方法二義性的問題。例如在以上的代碼中,MyClass同時繼承於BaseClassA和BaseClassB,假設MyClass調用一個叫derivedMethod方法,derivedMethod同時定義在BaseClassA和BaseClassB中,且Signature也完全相同,那么BaseClassA中的方法將被調用。如果BaseClassA中並沒有定義derivedMethod,而是BaseClassA的父類定義了這個方法的話,將會是BaseClassA的父類中derivedMethod被調用。總之,繼承方法搜索的路徑是先從左到右,在選定了一個BaseClass之后,將會一直沿着該BaseClass的繼承結構進行搜索,直至最頂端,然后再到另外一個一個BaseClass。
就先說着這么多了,對於Python中OO的特性將會在以后的Post中有進一步的講述。
[1] 方法二義性:由於一個類同時繼承於兩個或者多個父類,而在這些父類當中存在着signature完全相同的方法,那么編譯器將無法判斷子類將繼承哪個父類中的方法,從而導致方法二義性問題。
原文:http://www.cnblogs.com/perhaps/archive/2005/08/25/223005.html
========================================================
More reading,and english is important.
I'm Hongten
大哥哥大姐姐,覺得有用打賞點哦!多多少少沒關系,一分也是對我的支持和鼓勵。謝謝。
Hongten博客排名在100名以內。粉絲過千。
Hongten出品,必是精品。
E | hongtenzone@foxmail.com B | http://www.cnblogs.com/hongten
========================================================