一、多態
多態是指一類事物有多種形態,比如動物類,可以有貓,狗,豬等等。(一個抽象類有多個子類,因而多態的概念依賴于繼承)
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多態的資料請關注腳本之家其它相關文章!
您可能感興趣的文章:- 簡單了解Python多態與屬性運行原理
- Python 面向對象之封裝、繼承、多態操作實例分析
- Python面向對象之接口、抽象類與多態詳解
- Python面向對象編程之繼承與多態詳解
- Python 的類、繼承和多態詳解
- Python教程之Python多態的深層次理解