元类

这篇文章中总结了 Python 中面向对象的一些知识,在结尾处我们谈到了类对象和实例对象,实例对象是由类对象创建出来的,那么类对象是由什么创建出来的呢?这就是今天要总结的元类。

类对象的创建时机

类是什么时候创建的呢?我们可以进行一下验证:

class Test(object):
    print("我被创建啦!!")

运行代码:

我被创建啦!!

上例中,我们并没有做任何调用操作,Test 类内部的代码自动执行了,也就是说,当解释器执行到 class 关键字的时候,类就开始创建了。

动态创建类

我们也可以将类的创建放在条件判断中,以根据不同的条件创建类:

def createClass(flag):
    if flag:
        class Foo(object):
            print("Foo 被创建啦")
        return Foo
    else:
        class Boo(object):
            print("Boo 被创建啦")
        return Boo

createClass(True)

运行结果:

Foo 被创建啦

type

除了使用上面的条件判断,我们还可以使用 type 函数动态创建类。是的,除了判断变量的类型之外,我们还可以使用 type 函数动态创建类,使用方法如下:

类对象 = type("类名", 父类对象, 类属性)

其中父类对象是一个包含了父类的元组,类属性是一个包含了属性的字典。
看一下代码:

# 定义 __str__ 方法
def __str__(self):
    return "我是 Test 类"
# 动态创建 Test 类
Test = type("Test",(),{"__str__":__str__})
# 创建类实例
print(Test())

运行结果:

我是 Test 类

元类

上面我们使用 type 函数动态创建了一个类 Test,使用 type 可以创建类对象,我们称它为元类。所有类的元类都是 type(包括 type 自身)。

__class__

使用 __class__ 属性可以查看一个对象的创建者,类也是对象,因此我们也可以使用 __class__ 查看类的元类:

class Test(object):
    pass

print(Test().__class__)
print(Test.__class__)
print(type.__class__)
print(object.__class__)

运行效果:

<class '__main__.Test'>
<class 'type'>
<class 'type'>
<class 'type'>

可见,所有类的元类都是 type,包括 type 自身和 object

metaclass

除了在外部使用 type 动态创建类之外,我们还可以在类的内部指定其的创建方式,这就需要使用
metaclassmetaclass 是一个函数,在类创建时传入。该函数接受三个参数:

  • 类名
  • 父类
  • 类内部的属性

我们来看下实现代码:

def createClass(cls_name,cls_parent,cls_attr):
    attr = {}
    for k,v in cls_attr.items():
        if not k.startswith("__"):
            # 处理非私有属性
            # 将属性名转化为大写
            attr[k.upper()] = v
        else:
            # 处理非私有属性
            # 将属性名转化为大写
            attr["%s%s"%(cls_name,k.upper())] = v
    # 返回一个类,该类仍然由 type 创建
    return type(cls_name,cls_parent,attr)

class Test(object,metaclass = createClass):
    name = "Test"
    privateAttr = "10086"

# 创建类的实例
t = Test()

通过指定 metaclass,我们可以规定 Test 类的创建方法,在 createClass 方法中,我们对属性名进行大写转换,并进行了私有化的处理。最后返回一个使用 type 动态创建的类。
我们对类属性进行了大写转化,因此使用原始的属性名将获取不到数据:

...
# 创建类的实例
t = Test()
print(t.name)

运行结果:

Traceback (most recent call last):
  File "C:\Users\Charley\Desktop\py\py.py", line 21, in <module>
    print(t.name)
AttributeError: 'Test' object has no attribute 'name'

我们使用大写的属性名可以正常获取数据:

...
# 创建类的实例
t = Test()
print(t.NAME)

运行结果:

Test

Python2 中的 __metaclass__

上面的 metaclass 是 Python3 中的创建方式,在 Python2 中,需要改为 __metaclass__ 属性。
下面是 Python2 中的参考代码:

def createClass(cls_name,cls_parent,cls_attr):
    attr = {}
    for k,v in cls_attr.items():
        if not k.startswith("__"):
            # 处理非私有属性
            # 将属性名转化为大写
            attr[k.upper()] = v
        else:
            # 处理非私有属性
            # 将属性名转化为大写
            attr["%s%s"%(cls_name,k.upper())] = v
    # 返回一个类,该类仍然由 type 创建
    return type(cls_name,cls_parent,attr)

class Test:
    __metaclass__ = createClass()
    name = "Test"
    privateAttr = "10086"

# 创建类的实例
t = Test()
print(t.NAME)

总结

本文主要介绍了元类的知识点,下面是一些总结:

  • 类也是对象,叫做类对象,创建类对象的类叫做元类
  • 类可以动态创建
  • type 是所有类包括其自身的元类
  • 在类中可以使用 metaclass(Python3)和 __metaclass__(Python2)定义类的创建方式

完。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容

  • 什么是元类? 理解元类(metaclass)之前,我们先了解下Python中的OOP和类(Class) 面向对象全...
    时间之友阅读 324评论 0 0
  • 前言 第十二篇了,撸起袖子,就是干。 目录 一、Python 中类也是对象 在了解元类之前,我们先进一步理解 Py...
    GitHubClub阅读 758评论 0 7
  • 一元类 1类也是对象 在大多数编程语言中,类就是一组用来描述如何生成一个对象的代码段。在Python中这一点仍然成...
    五行缺觉阅读 1,048评论 0 1
  • 1. 类也是对象 在大多数编程语言中,类就是一组用来描述如何生成一个对象的代码段。在Python中这一点仍然成立:...
    ztfdeveloper阅读 295评论 0 0
  • 类也是对象,在理解元类之前,你需要先掌握Python中的类。Python中类的概念借鉴于Smalltalk,这显得...
    雲凌禹阅读 451评论 0 3