淺談python中的多態
一、多態
多態是指一類事物有多種形態,比如動物類,可以有貓,狗,豬等等。(一個抽象類有多個子類,因而多態的概念依賴於繼承)
import abc class Animal(metaclass=abc.ABCMeta): #同一類事物:動物 @abc.abstractmethod def talk(self): pass class Cat(Animal): #動物的形態之一:貓 def talk(self): print('say miaomiao') class Dog(Animal): #動物的形態之二:狗 def talk(self): print('say wangwang') class Pig(Animal): #動物的形態之三:豬 def talk(self): print('say aoao')
二、多態性
註意:多態與多態性是兩種概念
多態性是指具有不同功能的函數可以使用相同的函數名,這樣就可以用一個函數名調用不同內容的函數。在面向對象方法中一般是這樣表述多態性:向不同的對象發送同一條消息,不同的對象在接收時會產生不同的行為(即方法)。也就是說,每個對象可以用自己的方式去響應共同的消息。所謂消息,就是調用函數,不同的行為就是指不同的實現,即執行不同的函數。
import abc class Animal(metaclass=abc.ABCMeta): #同一類事物:動物 @abc.abstractmethod def talk(self): pass class Cat(Animal): #動物的形態之一:貓 def talk(self): print('say miaomiao') class Dog(Animal): #動物的形態之二:狗 def talk(self): print('say wangwang') class Pig(Animal): #動物的形態之三:豬 def talk(self): print('say aoao') c = Cat() d = Dog() p = Pig() def func(obj): obj.talk() func(c) func(d) func(p) ------------------------------ >>> say miaomiao >>> say wangwang >>> say aoao
綜上可以說,多態性是 : 一個接口,多種實現
多態性的好處:
- 增加瞭程序的靈活性,以不變應萬變,不論對象千變萬化,使用者都是同一種形式去調用,如func(obj)
- 增加瞭程序額可擴展性,通過繼承animal類創建瞭一個新的類,使用者無需更改自己的代碼,還是用func(obj)去調用
三、鴨子類型
調用不同的子類將會產生不同的行為,而無須明確知道這個子類實際上是什麼,這是多態的重要應用場景。而在python中,因為鴨子類型(duck typing)使得其多態不是那麼酷。
鴨子類型是動態類型的一種風格。在這種風格中,一個對象有效的語義,不是由繼承自特定的類或實現特定的接口,而是由”當前方法和屬性的集合”決定。這個概念的名字來源於由James Whitcomb Riley提出的鴨子測試,“鴨子測試”可以這樣表述:“當看到一隻鳥走起來像鴨子、遊泳起來像鴨子、叫起來也像鴨子,那麼這隻鳥就可以被稱為鴨子。”
在鴨子類型中,關註的不是對象的類型本身,而是它是如何使用的。例如,在不使用鴨子類型的語言中,我們可以編寫一個函數,它接受一個類型為”鴨子”的對象,並調用它的”走”和”叫”方法。在使用鴨子類型的語言中,這樣的一個函數可以接受一個任意類型的對象,並調用它的”走”和”叫”方法。如果這些需要被調用的方法不存在,那麼將引發一個運行時錯誤。任何擁有這樣的正確的”走”和”叫”方法的對象都可被函數接受的這種行為引出瞭以上表述,這種決定類型的方式因此得名。
鴨子類型通常得益於不測試方法和函數中參數的類型,而是依賴文檔、清晰的代碼和測試來確保正確使用。
Duck typing 這個概念來源於美國印第安納州的詩人詹姆斯·惠特科姆·萊利(James Whitcomb Riley,1849- 1916)的詩句:”When I see a bird that walks like a duck and swims like a duck and quacks like a duck, I call that bird a duck.”
先上代碼,也是來源於網上很經典的案例:
class Duck(): def walk(self): print('I walk like a duck') def swim(self): print('i swim like a duck') class Person(): def walk(self): print('this one walk like a duck') def swim(self): print('this man swim like a duck')
可以很明顯的看出,Person類擁有跟Duck類一樣的方法,當有一個函數調用Duck類,並利用到瞭兩個方法walk()和swim()。我們傳入Person類也一樣可以運行,函數並不會檢查對象的類型是不是Duck,隻要他擁有walk()和swim()方法,就可以正確的被調用。
再舉例,如果一個對象實現瞭__getitem__方法,那python的解釋器就會把它當做一個collection,就可以在這個對象上使用切片,獲取子項等方法;如果一個對象實現瞭__iter__和next方法,python就會認為它是一個iterator,就可以在這個對象上通過循環來獲取各個子項。
以上就是淺談python中的多態的詳細內容,更多關於python多態的資料請關註WalkonNet其它相關文章!