亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Python深入淺出分析元類(lèi)

 更新時(shí)間:2022年07月21日 09:20:06   作者:Flyme?awei  
在Python里一切都是對(duì)象(object),基本數(shù)據(jù)類(lèi)型,如數(shù)字,字符串,函數(shù)都是對(duì)象。對(duì)象可以由類(lèi)(class)進(jìn)行創(chuàng)建。那么既然一切都是對(duì)象,那么類(lèi)是對(duì)象嗎?是的,類(lèi)也是對(duì)象,那么又是誰(shuí)創(chuàng)造了類(lèi)呢?答案也很簡(jiǎn)單,也是類(lèi),一個(gè)能創(chuàng)作類(lèi)的類(lèi),稱(chēng)之為(type)元類(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)文章

最新評(píng)論