儘管Python在Function Programming中有著其他語言難以企及的優勢,但是我們也不要忘了Python也是一門OO語言哦。因此我們關注Python在FP上的優勢的同時,也得了解Python在OO方面的特性。
要討論Python的OO特性,了解Python中的Class自然是首當其衝了。在Python中定義class和創建物件實例都很簡單,具體程式碼如下:
class GrandPa: def __init__(self): print('I\'m GrandPa') class Father(GrandPa): def __init__(self): print('I\'m Father!') class Son(Father): """A simple example class""" i = 12345 def __init__(self): print('这是构造函数,son') def sayHello(self): return 'hello world' if __name__ == '__main__': son = Son() # 类型帮助信息 print('类型帮助信息: ',Son.__doc__) #类型名称 print('类型名称:',Son.__name__) #类型所继承的基类 print('类型所继承的基类:',Son.__bases__) #类型字典 print('类型字典:',Son.__dict__) #类型所在模块 print('类型所在模块:',Son.__module__) #实例类型 print('实例类型:',Son().__class__)
運行情況:
Python 3.3.2 (v3.3.2:d047928ae3f6, May 16 2013, 00:d03:43) [MSC vMSC. 1600 32 bit (Intel)] on win32
Type "copyright", "credits" or "license()" for more information.
>>> ================== ================ RESTART ================================
>>>
這是建構函式,son
型別幫助資訊: A simple example class
型別名稱: Son
型別所繼承的基底類別: (__
, '__doc__': 'A simple example class', '__init__':
實例類型:
#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完全相同的方法,那麼編譯器將無法判斷子類別將繼承哪個父類別中的方法,從而導致方法二義性問題