Python深入淺出分析元類(lèi)
一、類(lèi)和對(duì)象
Python屬于動(dòng)態(tài)類(lèi)型的語(yǔ)言,而動(dòng)態(tài)語(yǔ)言和靜態(tài)語(yǔ)言最大的不同,就是函數(shù)和類(lèi)的定義,不是編譯時(shí)創(chuàng)建的,而是運(yùn)行時(shí)動(dòng)態(tài)創(chuàng)建的,比方說(shuō)我們要定義一個(gè) Person
的class,就寫(xiě)一個(gè) Person.py
模塊:
# -*- coding: utf-8 -*- # 文件名 : Person.py class Person(object): def say(self, s='元類(lèi)'): print('今日學(xué)習(xí):%s' % s) if __name__ == '__main__': p = Person() p.say()
當(dāng)Python解釋器載入 Person
模塊時(shí),就會(huì)依次執(zhí)行該模塊的所有語(yǔ)句,執(zhí)行結(jié)果就是動(dòng)態(tài)創(chuàng)建出一個(gè) Person
的class對(duì)象,測(cè)試如下,(注意,是引入一個(gè)Person.py腳本)
# 文件名 :測(cè)試.py # 引入模塊,需要模塊的路徑 from Person import Person # 創(chuàng)建一個(gè)Person類(lèi)的實(shí)例 p = Person() p.say('Python中的元類(lèi)') # 調(diào)用say方法 # 今日學(xué)習(xí):Python中的元類(lèi) print(type(p)) # 把實(shí)例 p 的類(lèi)型打印出來(lái) # <class 'say.Person'> print(type(Person)) # Person 類(lèi)的類(lèi)型打印出來(lái) # <class 'type'>
這里是用來(lái) type()
函數(shù),可以查看一個(gè)類(lèi)型或變量,的類(lèi)型, Person
是一個(gè)class,它的類(lèi)型就是 type
,而 p
是一個(gè)實(shí)例,它的類(lèi)型就是class Person
。
二、type類(lèi)
我們說(shuō)class的定義是運(yùn)行時(shí)動(dòng)態(tài)創(chuàng)建的,而創(chuàng)建class的方法就是使用 type()
函數(shù)
type()
函數(shù)既可以返回一個(gè)對(duì)象的類(lèi)型,又可以創(chuàng)建出新的類(lèi)型,比如,我們可以通過(guò) type()
函數(shù)創(chuàng)建出Person 類(lèi),而無(wú)需通過(guò) class Person(object)...
的定義,此時(shí)type的第二種用法,我們只要type傳object是可以返回該對(duì)象類(lèi)型的,但是當(dāng)我們的type存在三位參數(shù)存在時(shí)
# type 類(lèi) class type(object): """ type(object_or_name, bases, dict) type(object) -> the object's type type(name, bases, dict) -> a new type """ #參數(shù)介紹 """ type(object_or_name, bases, dict) object-or-name -- 對(duì)象或類(lèi)的名稱(chēng)。 bases -- 基類(lèi)的元組。 dict -- 字典,類(lèi)內(nèi)定義的命名空間變量。 """ # 返回新的類(lèi)型對(duì)象。
# -*- coding: utf-8 -*- def fun(self, s): print('hello'+s) Hello = type('hello', (object,), dict(func=fun)) if __name__ == '__main__': h = Hello() h.func(' word') # hello word print(type(h)) # <class '__main__.hello'> print(type(Hello)) # <class 'type'>
我們通過(guò) type()
函數(shù)創(chuàng)建的類(lèi)和直接寫(xiě)class是完全一樣的,因?yàn)镻ython解釋器遇到class定義時(shí),僅僅是掃描一下class定義的語(yǔ)法,然后調(diào)用 type()
函數(shù)創(chuàng)建出class
在正常情況下,我們都用 class Xxx...
來(lái)定義類(lèi),但是, type()
函數(shù)也允許我們動(dòng)態(tài)創(chuàng)建出類(lèi)來(lái),也就是說(shuō),動(dòng)態(tài)語(yǔ)言本身支持運(yùn)行期動(dòng)態(tài)創(chuàng)建類(lèi),這和靜態(tài)語(yǔ)言有非常大的不同,要在靜態(tài)語(yǔ)言運(yùn)行期創(chuàng)建類(lèi),必須構(gòu)造源代碼字符串再調(diào)用編譯器,或者借助一些工具生成字節(jié)碼實(shí)現(xiàn),本質(zhì)上都是動(dòng)態(tài)編譯,會(huì)非常復(fù)雜
三、元類(lèi)Metaclass
除了使用 type()
動(dòng)態(tài)創(chuàng)建類(lèi)以外,要控制類(lèi)的創(chuàng)建行為,還可以使用metaclass
,也就是元類(lèi)
當(dāng)我們定義了類(lèi)以后,就可以根據(jù)這個(gè)類(lèi)創(chuàng)建出實(shí)例,所以:先定義類(lèi),然后創(chuàng)建實(shí)例
但是如果我們想創(chuàng)建出類(lèi)呢?那就必須根據(jù)metaclass創(chuàng)建出類(lèi),所以:先定義metaclass
,然后創(chuàng)建類(lèi),元類(lèi)就是用來(lái)創(chuàng)建這些類(lèi)(對(duì)象)的,元類(lèi)就是類(lèi)的類(lèi)
我們先看一個(gè)簡(jiǎn)單的例子,這個(gè)metaclass
可以給自定義的類(lèi)增加一個(gè) add
方法定義 ListMetaclass
,按照默認(rèn)習(xí)慣,metaclass的類(lèi)名總是以Metaclass結(jié)尾,以便清楚地表示這是一個(gè)metaclas。
四、自定義一個(gè)元類(lèi)
class UpperMetaclass(type): def __new__(mcs, class_name, class_parents, class_attrs): new_attrs = {} for name, value in class_attrs.items(): if not name.startswith('__'): # 判斷是否為非私有屬性 new_attrs[name.upper()] = value # 直接調(diào)用type 來(lái)創(chuàng)建一個(gè)類(lèi) return type.__new__(mcs, class_name, class_parents, class_attrs) # 測(cè)試 class Emp(object, metaclass=UpperMetaclass): name = '張三' acl = 500 if __name__ == '__main__': print(hasattr(Emp, 'name')) # 判斷Emp中是否有名字為name print(hasattr(Emp, 'NAME')) # 判斷Emp中是否有名字為NAME
到此這篇關(guān)于Python深入淺出分析元類(lèi)的文章就介紹到這了,更多相關(guān)Python元類(lèi)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Python?OpenCV識(shí)別行人入口進(jìn)出人數(shù)統(tǒng)計(jì)
本文主要介紹了Python?OpenCV識(shí)別行人入口進(jìn)出人數(shù)統(tǒng)計(jì),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧<BR>2023-01-01Python bisect模塊原理及常見(jiàn)實(shí)例
這篇文章主要介紹了Python bisect模塊原理及常見(jiàn)實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-06-06Python 網(wǎng)絡(luò)編程起步(Socket發(fā)送消息)
現(xiàn)在開(kāi)始學(xué)習(xí)網(wǎng)絡(luò)編程,先從簡(jiǎn)單的UDP協(xié)議發(fā)送消息開(kāi)始。我們需要有接受消息的服務(wù)端程序(Server.py)和發(fā)送消息的客戶端程序(Client)。2008-09-09python?aeon庫(kù)進(jìn)行時(shí)間序列算法預(yù)測(cè)分類(lèi)實(shí)例探索
這篇文章主要介紹了python?aeon庫(kù)進(jìn)行時(shí)間序列算法預(yù)測(cè)分類(lèi)實(shí)例探索,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-02-02python機(jī)器學(xué)習(xí)GCN圖卷積神經(jīng)網(wǎng)絡(luò)原理解析
這篇文章主要為大家介紹了GCN圖卷積神經(jīng)網(wǎng)絡(luò)原理及代碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-05-05python 兩個(gè)一樣的字符串用==結(jié)果為false問(wèn)題的解決
這篇文章主要介紹了python 兩個(gè)一樣的字符串用==結(jié)果為false問(wèn)題的解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-03-03Python利用Selenium實(shí)現(xiàn)彈出框的處理
經(jīng)常出現(xiàn)在網(wǎng)頁(yè)上的基于JavaScript實(shí)現(xiàn)的彈出框有三種,分別是?alert、confirm、prompt?。本文主要是學(xué)習(xí)如何利用selenium處理這三種彈出框,感興趣的可以了解一下2022-06-06