国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > Python > 正文

Python編程中對super函數的正確理解和用法解析

2019-11-25 16:40:04
字體:
來源:轉載
供稿:網友

當在子類需要調用父類的方法時,在python2.2之前,直接用類名調用類的方法,即非綁定的類方法,并把自身對象self作參數傳進去。

class A(object):   def say(self):     print 'I am A'  class B(A):   def say(self):     print 'I am B'     A.say(self)  b = B() b.say() 

輸出

I am BI am A

這樣運作挺好,不過有個問題,當父類改了名字時,就要把這些顯式調用父類的一個個更正,子類和父類耦合比較高。
于是python2.2后就推出了super()函數來避免硬編碼,不用關心父類名叫什么。
使用super()函數,上面的代碼可以寫成如下。

class B(A):   def say(self):     print 'I am B'     super(B,self).say() 

python3.0后,又做了改良,super()函數不用傳參數,即上面的那行代碼直接super().say()就行了。

需要注意的問題:

  • super只能用在新式類中。
  • super在多重繼承有問題,如果子類繼承多個父類,那么super調用第一個父類的方法。
  • 不要混用這兩種調用父類方法的方案,要么都用非綁定的類方法,要么都用super。不然可能導致沒被調用或者被調用多次。

BUT:
不要一說到 super 就想到父類!super 指的是 MRO 中的下一個類!
一說到 super 就想到父類這是初學者很容易犯的一個錯誤,也是我當年犯的錯誤。

def super(cls, inst):  mro = inst.__class__.mro()  return mro[mro.index(cls) + 1]

兩個參數 cls 和 inst 分別做了兩件事:
1. inst 負責生成 MRO 的 list
2. 通過 cls 定位當前 MRO 中的 index, 并返回 mro[index + 1]
這兩件事才是 super 的實質,一定要記住!
MRO 全稱 Method Resolution Order,它代表了類繼承的順序。

舉個例子:

class Root(object):  def __init__(self):    print("this is Root")class B(Root):  def __init__(self):    print("enter B")    # print(self) # this will print <__main__.D object at 0x...>    super(B, self).__init__()    print("leave B")class C(Root):  def __init__(self):    print("enter C")    super(C, self).__init__()    print("leave C")class D(B, C):  passd = D()print(d.__class__.__mro__)

輸出

enter Benter Cthis is Rootleave Cleave B(<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.Root'>, <type 'object'>)

知道了 super 和父類其實沒有實質關聯之后,我們就不難理解為什么 enter B 下一句是 enter C 而不是 this is Root(如果認為 super 代表“調用父類的方法”,會想當然的認為下一句應該是this is Root)。流程如下,在 B 的 __init__ 函數中:

super(B, self).__init__()

首先,我們獲取 self.__class__.__mro__,注意這里的 self 是 D 的 instance 而不是 B 的

(<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.Root'>, <type 'object'>)

然后,通過 B 來定位 MRO 中的 index,并找到下一個。顯然 B 的下一個是 C。于是,我們調用 C 的 __init__,打出 enter C。

順便說一句為什么 B 的 __init__ 會被調用:因為 D 沒有定義 __init__,所以會在 MRO 中找下一個類,去查看它有沒有定義 __init__,也就是去調用 B 的 __init__。

其實這一切邏輯還是很清晰的,關鍵是理解 super 到底做了什么。


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 银川市| 宜昌市| 滁州市| 麟游县| 丹阳市| 六安市| 宁远县| 疏勒县| 梁平县| 沂水县| 东莞市| 雷山县| 来安县| 巴里| 涪陵区| 彩票| 东丽区| 义乌市| 弋阳县| 陆川县| 个旧市| 清苑县| 乌什县| 云阳县| 修水县| 锡林郭勒盟| 民权县| 行唐县| 蓬溪县| 永泰县| 龙陵县| 武山县| 宁海县| 莱州市| 东海县| 福贡县| 宜章县| 鹿邑县| 锡林郭勒盟| 黄大仙区| 上饶县|