python元类

在大多数编程语言中,类就是一组用来描述如何生成一个对象的代码段。在Python中这一点仍然成立类也是对象。但是,Python中的类还远不止如此。类同样也是一种对象。是的,没错,就是对象。只要你使用关键字class,Python解释器在执行的时候就会创建一个对象。类是一组用来描述如何生成一个对象的代码段。

class Dog():
    pass


d1 = Dog()
print(type(d1))
print(type(Dog))
print(type(type))

结果如下:



d1是有Dog创建的,所以他是Dog的类。Dog是type的类,而type还是type的类。所以说类也是对象。因为类也是对象,就像其他任何对象一样,可以在运行时动态的创建。
首先可以在函数中创建类,使用class关键字即可。

def CreatClass(name):
    if name ==  'foo':
        class Foo(object):
            pass
        return Foo
    else:
        class Koo(object):
            pass
        return Koo

t1 = CreatClass('foo')
print(t1)
t2 = CreatClass('xx')
print(t2)

结果如下:



这显然不够动态,因为仍然需要自己编写整个类的代码。由于类是由type创建的,type可以接受一个类的描述作为参数,然后返回一个类。因此,可以直接用type手动创建类。

'''
使用type创建类
type('类名',(继承的父类(可以为空)),{属性(名称和值)或者方法(方法名和方法)})
'''
Person = type('Person',(),{})
print(Person)
print(dir(Person))

People = type('People',(object,),{'name':'小明','age':'18'})
print(People)
print(dir(People))

def show(self):
    print('姓名:%s 年龄:%s'%(self.name,self.age))

Dog = type('Dog',(),{'name':'小白','age':'3','show':show})
d1 = Dog()
print(dir(d1))
d1.show()

结果如下:


dir()是查看类的所以属性和方法,截图截不完,后看到后两个dir的显示会有name和age的属性,同时最后一个还有一个show。需要注意的是元组中添加父类时,如果只有一个,需要加逗号。
元类就是用来创建类的“东西”。python中所有的东西(包括整数、函数、字符串等等)都是对象。
MyClass = MetaClass() #使用元类创建一个对象,这个对象称为‘类’
MyObject = MyClass() #使用‘类’来创建实例对象
以上代码可以这样写:
Myclass = type(‘Myclass’,(),{})
函数type实际上是一个元类,type就是python在背后用来创建所有类的元类。使用class可以查看属性

可以看到,a属于字符串类型,而字符串类型又属于type类型。如果再加一个class会发现type还是属于type。
在定义一个类时,可以添加metaclass属性。
创建方案一:
class Foo ()
  methclass = something
  pass
对比创建方案二:
class Foo(Father):
  pass
如果使用创建方案一,python就会用元类来创建类Foo。首先,写下class Foo(),但是类Foo还没有在内存中创建。python会在类的定义中寻找metaclass属性,如果找到了,Python就会用它来创建类Foo,如果没有找到,就会用内建的type来创建这个类。如果选用创建方案二,python将做如下操作:
1、Foo中有metaclass这个属性吗?如果有,Python会通过metaclass创建一个名字为Foo的类(对象)
2、如果Python没有找到metaclass,它会继续在Father(父类)中寻找metaclass属性,并尝试做和前面同样的操作。
3、如果Python在任何父类中都找不到metaclass,它就会在模块层次中去寻找metaclass,并尝试做同样的操作。
4、如果还是找不到metaclass,Python就会用内置的type来创建这个类对象。
那可以在metaclass中添加什么代码?比如,你决定在你的模块里所有的类的属性都应该是大写形式。有好几种方法可以办到,但其中一种就是通过在模块级别设定metaclass。采用这种方法,这个模块中的所有类都会通过这个元类来创建,我们只需要告诉元类把所有的属性都改成大写形式就万事大吉了。

def upper_attr(future_class_name, future_class_parents, future_class_attr):
    print(future_class_name)          #类名
    print(future_class_parents)       #父类
    print(future_class_attr)          #属性
    #遍历属性字典,把不是__开头的属性名字变为大写
    newAttr = {}
    for name,value in future_class_attr.items():
        if not name.startswith("__"):
            newAttr[name.upper()] = value

    #调用type来创建一个类
    return type(future_class_name, future_class_parents, newAttr)


class Foo(object, metaclass = upper_attr):
    bar = 'bip'
    def haha(self):
        pass

print(hasattr(Foo, 'bar'))
# 输出: False
print(hasattr(Foo, 'BAR'))
# 输出:True
print(hasattr(Foo, 'haha'))
# 输出: False
print(hasattr(Foo, 'HAHA'))
# 输出:True
f = Foo()
print(f.BAR)
# 输出:'bip'

结果如下:



这次用一个class来当元类。

class UpperAttrMetaClass(type):
    # __new__ 是在__init__之前被调用的特殊方法
    # __new__是用来创建对象并返回之的方法
    # 而__init__只是用来将传入的参数初始化给对象
    # 你很少用到__new__,除非你希望能够控制对象的创建
    # 这里,创建的对象是类,我们希望能够自定义它,所以我们这里改写__new__
    def __new__(cls, future_class_name, future_class_parents, future_class_attr):
        # 遍历属性字典,把不是__开头的属性名字变为大写
        newAttr = {}
        for name, value in future_class_attr.items():
            if not name.startswith("__"):
                newAttr[name.upper()] = value
        # 方法3:使用super方法
        return super(UpperAttrMetaClass, cls).__new__(cls, future_class_name, future_class_parents, newAttr)


class Foo(object, metaclass=UpperAttrMetaClass):
    bar = 'bip'

    def haha(self):
        pass

print(hasattr(Foo, 'bar'))
# 输出: False
print(hasattr(Foo, 'BAR'))
# 输出:True
print(hasattr(Foo, 'haha'))
# 输出: False
print(hasattr(Foo, 'HAHA'))
# 输出:True
f = Foo()
print(f.BAR)
# 输出:'bip'

结果如下:



以上就是关于元类的总结。

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

推荐阅读更多精彩内容

  • 类也是对象,在理解元类之前,你需要先掌握Python中的类。Python中类的概念借鉴于Smalltalk,这显得...
    雲凌禹阅读 445评论 0 3
  • 1. 类也是对象 在大多数编程语言中,类就是一组用来描述如何生成一个对象的代码段。在Python中这一点仍然成立:...
    ztfdeveloper阅读 295评论 0 0
  • 1.元类 1.1.1类也是对象 在大多数编程语言中,类就是一组用来描述如何生成一个对象的代码段。在Python中这...
    TENG书阅读 1,253评论 0 3
  • 声明:本文仅限于简书发布,其他第三方网站均为盗版,原文地址: python元类浅析 在 python 的新式类中,...
    liuliqiang阅读 456评论 0 4
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,585评论 18 139