你可能知道 Python 里面的魔法函數(shù),比如 __add__ 和 __sub__ 代表 + - 運算符,表示 obj +/- something,但你可能不知道還有一個 __radd__,__rsub__ 函數(shù),可以表示 something +/- obj。
這 7 個問題,我是有收獲的,整理如下:
1、反射算術(shù)運算符
你可能知道 Python 里面的魔法函數(shù),比如 __add__? 和 __sub__? 代表 + - 運算符,表示 obj +/- something,但你可能不知道還有一個 __radd__,__rsub__ 函數(shù),可以表示 something +/- obj。
舉例如下:
class Dog:
def __add__(self, other):
return "from __add__"
def __radd__(self, other):
return "from __radd__"
dog = Dog()
print(dog + 1) # from __add__
print(1 + dog) # from __radd__
2、__getattr__ vs __getattribute__
__getattr__? 魔術(shù)方法只有在我們試圖獲取不存在的屬性時才會被調(diào)用,__getattribute__ 在每次我們嘗試訪問屬性時都會被調(diào)用。
代碼如下:
class Dog:
def __init__(self, name, age):
self.name = name
self.age = age
def __getattr__(self, key):
return f"{key} not found"
dog = Dog("taidi", 5)
print(dog.name) # taidi
print(dog.age) # 5
print(dog.breed) # breed not found
class Dog:
def __init__(self, name, age):
self.name = name
self.age = age
def __getattribute__(self, key):
return f"{key} not found"
dog = Dog("taidi", 5)
print(dog.name) # name not found
print(dog.age) # age not found
print(dog.breed) # breed not found
3、super().__init__() 的另一種寫法
class Animal:
def __init__(self, name, age):
self.name = name
self.age = age
class Dog(Animal):
def __init__(self, name, age, breed):
super().__init__(name, age)
self.breed = breed
等價于:
class Animal:
def __init__(self, name, age):
self.name = name
self.age = age
class Dog(Animal):
def __init__(self, name, age, breed):
Animal.__init__(self, name, age)
self.breed = breed
請注意,Animal.__init__(self, name, age) 不能少了 self 參數(shù)。
4、檢查子類的方法
class Animal: pass
class Dog(Animal): pass
class Cat(Animal): pass
class GermanSheperd(Dog): pass
print(Animal.__subclasses__())
# [<class '__main__.Dog'>, <class '__main__.Cat'>]
不過,.__subclasses__() 只能檢查直接子類。
5、多重集成時,同名函數(shù),子類用的是哪一個?
class A:
def test(self):
print("A")
class B:
def test(self):
print("B")
class C(A, B):
pass
C().test() # A
A 和 B 都有 test 方法,那么 C 到底集成了哪一個呢?在 Python 中,最左邊的類優(yōu)先。
在這里,A 是最左邊的父類,因此 A 的 test 方法被集成。
多充繼承讓人困惑,不用為好。
6 __invert__ 魔法函數(shù)
class Dog:
def __invert__(self):
return "test"
dog = Dog()
print(~dog) # test
~ 運算符代表“按位非”,通常用于反轉(zhuǎn)內(nèi)容。一個更有意義的例子如下:
class Coordinate:
def __init__(self, x, y):
self.x = x
self.y = y
def __str__(self):
return f"({self.x}, {self.y})"
def __invert__(self):
return Coordinate(-self.x, -self.y)
a = Coordinate(3, 4)
b = ~a
print(a, b) # (3, 4) (-3, -4)
7、不使用 class 來創(chuàng)建類
def init(self, name, age):
self.name = name
self.age = age
def bark(self):
print("woof")
Dog = type("Dog", (), {"__init__":init, "bark":bark})
dog = Dog("taidi", 10)
print(dog.name)
print(dog.age)
# taidi
# 10
在這里,我們將 3 個參數(shù)傳遞給 type 以創(chuàng)建我們的類。
第一個參數(shù) __name__? 是類的名稱 第二個參數(shù) __bases__? 是一個包含父類的元組 第三個參數(shù) __dict__ 是一個包含屬性和方法的字典。
等價于:
class Dog:
def __init__(self, name, age):
self.name = name
self.age = age
def bark(self):
print("woof")