您好,登錄后才能下訂單哦!
這篇文章主要介紹“Python元編程知識點有哪些”的相關知識,小編通過實際案例向大家展示操作過程,操作方法簡單快捷,實用性強,希望這篇“Python元編程知識點有哪些”文章能幫助大家解決問題。
首先說,Python中一切皆對象,老生常談。還有,Python提供了許多特殊方法、元類等等這樣的“元編程”機制。像給對象動態添加屬性方法之類的,在Python中根本談不上是“元編程”,但在某些靜態語言中卻是需要一定技巧的東西。我們來談些Python程序員也容易被搞糊涂的東西。
我們先來把對象分分層次,通常我們知道一個對象有它的類型,老早以前Python就將類型也實現為對象。這樣我們就有了實例對象和類對象。這是兩個層次。稍有基礎的讀者就會知道還有元類這個東西的存在,簡言之,元類就是“類”的“類”,也就是比類更高層次的東西。這又有了一個層次。還有嗎?
如果我們換個角度,不用非得和之前的三個層次使用同樣的標準。我們再來區分兩個東西:ImportTime和RunTime,它們之間也并非界限分明,顧名思義,就是兩個時刻,導入時和運行時。
當一個模塊被導入時,會發生什么?在全局作用域的語句(非定義性語句)被執行。函數定義呢?一個函數對象被創建,但其中的代碼不會被執行。類定義呢?一個類對象被創建,類定義域的代碼被執行,類的方法中的代碼自然也不會被執行。
執行時呢?函數和方法中的代碼會被執行。當然你要先調用它們。
所以我們可以說元類和類是屬于ImportTime的,import一個模塊之后,它們就會被創建。實例對象屬于RunTime,單import是不會創建實例對象的。不過話不能說的太絕對,因為如果你要是在模塊作用域實例化類,實例對象也是會被創建的。只不過我們通常把它們寫在函數里面,所以這樣劃分。
如果你想控制產生的實例對象的特性該怎么做?太簡單了,在類定義中重寫__init__方法。那么我們要控制類的一些性質呢?有這種需求嗎?當然有!
經典的單例模式,大家都知道有很多種實現方式。要求就是,一個類只能有一個實例。
最簡單的實現方法是這樣的
class _Spam:
def __init__(self):
print("Spam!!!")
_spam_singleton =None
def Spam():
global _spam_singleton
if _spam_singleton is not None:
return _spam_singleton
else:
_spam_singleton = _Spam()
return _spam_singleton
工廠模式,不太優雅。我們再來審視一下需求,要一個類只能有一個實例。我們在類中定義的方法都是實例對象的行為,那么要想改變類的行為,就需要更高層次的東西。元類在這個時候登場在合適不過了。前面說過,元類是類的類。也就是說,元類的__init__
方法就是類的初始化方法。 我們知道還有__call__
這個東西,它能讓實例像函數那樣被調用,那么元類的這個方法就是類在被實例化時調用的方法。
代碼就可以寫出來了:
class Singleton(type):
def __init__(self, *args, **kwargs):
self._instance = None
super().__init__(*args, **kwargs)
def __call__(self, *args, **kwargs):
if self._instance is None:
self._instance = super().__call__(*args, **kwargs)
return self._instance
else:
return self._instance
class Spam(metaclass=Singleton):
def __init__(self):
print("Spam!!!")
主要有兩個地方和一般的類定義不同,一是Singleton的基類是type,一是Spam定義的地方有一個metaclass=Singleton。type是什么?它是object的子類,object是它的實例。也就是說,type是所有類的類,也就是最基本的元類,它規定了一些所有類在產生時需要的一些操作。所以我們的自定義元類需要子類化type。同時type也是一個對象,所以它又是object的子類。有點不太好理解,大概知道就可以了。
我們再來說說裝飾器。大多數人認為裝飾器是Python里面最難理解的概念之一。其實它不過就是一個語法糖,理解了函數也是對象之后。就可以很輕易的寫出自己的裝飾器了。
from functools import wraps
def print_result(func):
@wraps(func)
def wrappper(*args, **kwargs):
result = func(*args, **kwargs)
print(result)
return result
return wrappper
@print_result
def add(x, y):
return x + y
#相當于:
#add = print_result(add)
add(1, 3)
這里我們還用到了一個裝飾器@wraps,它是用來讓我們返回的內部函數wrapper和原來的函數擁有相同的函數簽名的,基本上我們在寫裝飾器時都要加上它。
在注釋里寫了,@decorator這樣的形式等價于func=decorator(func),理解了這一點,我們就可以寫出更多種類的裝飾器。比如類裝飾器,以及將裝飾器寫成一個類。
def attr_upper(cls):
for attrname,value in cls.__dict__.items():
if isinstance(value,str):
if not value.startswith('__'):
setattr(cls,attrname,bytes.decode(str.encode(value).upper()))
return cls
@attr_upper
class Person:
sex = 'man'
print(Person.sex) # MAN
注意普通的裝飾器和類裝飾器實現的不同點。
如果我們想讓某一些類擁有某些相同的特性,或者說可以實現在類定義對其的控制,我們可以自定義一個元類,然后讓它成為這些類的元類。如果我們想讓某一些函數擁有某些相同的功能,又不想把代碼復制粘貼一遍,我們可以定義一個裝飾器。那么,假如我們想讓實例的屬性擁有某些共同的特點呢?有人可能會說可以用property,當然可以。但是這些邏輯必須在每個類定義的時候都寫一遍。如果我們想讓這些類的實例的某些屬性都有相同的特點的話,就可以自定義一個描述符類。
這里我們給出一些例子
class TypedField:
def __init__(self, _type):
self._type = _type
def __get__(self, instance, cls):
if instance is None:
return self
else:
return getattr(instance, self.name)
def __set_name__(self, cls, name):
self.name = name
def __set__(self, instance, value):
if not isinstance(value, self._type):
raise TypeError('Expected' + str(self._type))
instance.__dict__[self.name] = value
class Person:
age = TypedField(int)
name = TypedField(str)
def __init__(self, age, name):
self.age = age
self.name = name
jack = Person(15, 'Jack')
jack.age = '15' # 會報錯
在這里面有幾個角色,TypedField
是一個描述符類,的屬性Person
是描述符類的實例,看似描述符是作為Person,也就是類的屬性而不是實例屬性存在的。但實際上,一旦Person的實例訪問了同名的屬性,描述符就會起作用。需要注意的是,在Python3.5
及之前的版本中,是沒有__set_name__
這個特殊方法的,這意味著如果你想要知道在類定義中描述符被起了一個什么樣的名字,是需要在描述符實例化時顯式傳遞給它的,也就是需要多一個參數。不過在Python3.6中,這個問題得到了解決,只需要在描述符類定義中重寫__set_name__
這個方法就好了。還需要注意的是__get__
的寫法,基本上對instance的判斷是必需的,不然會報錯。原因也不難理解,就不細說了。
在Python3.6中,我們可以通過實現__init_subclass__特殊方法,來自定義子類的創建,這樣我們就可以在某些情況下擺脫元類這個討厭的東西。
class PluginBase:
subclasses = []
def __init_subclass__(cls, **kwargs):
super().__init_subclass__(**kwargs)
cls.subclasses.append(cls)
class Plugin1(PluginBase):
pass
class Plugin2(PluginBase):
pass
關于“Python元編程知識點有哪些”的內容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業相關的知識,可以關注億速云行業資訊頻道,小編每天都會為大家更新不同的知識點。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。