Python 使用元类type创建类对象常见应用详解
Python  /  管理员 发布于 7年前   139
本文实例讲述了Python 使用元类type创建类对象。分享给大家供大家参考,具体如下:
type("123")
可以查看变量的类型;同时 type("类名",(父类),{类属性:值,类属性2:值})
可以创建一个类。
在Python中不建议一个函数具有不同的功能(重载);type()具有不同的功能是为了兼容之前的版本。
类可以创建实例对象,类对象是由元类创建的。 (元类创建类,类创建实例对象)
type就是元类(type本质上就是一个类)
demo.py(用元类type创建类):
# 通过class关键字创建类class MyClass1(object): name = "张三" # 类属性 (所有实例对象共用) age = 23# 通过type创建类。 type()返回的是创建的类对象的引用。Test2 = type("MyClass2",(object,),{"name":"张三","age":23}) # Test2是MyClass2类的引用,一般变量名和类名保持一致。print(Test2()) # <__main__.MyClass2 object at 0x7fa05a4ca9e8>
demo.py(用type创建带有方法的类):
# 实例方法def print_b(self): print(self.num)# 静态方法@staticmethoddef print_static(): print("----haha-----")# 类方法@classmethoddef print_class(cls): print(cls.num)# 用type创建类B = type("B", (object,), {"num":100, "print_b": print_b, "print_static": print_static, "print_class": print_class})b = B()b.print_b() # 100b.print_static() # ----haha-----b.print_class() # 100
元类的应用
在定义一个类的时候可以为其指定__metaclass__属性(指定创建该类的元类),默认使用type元类创建类对象。
通过指定自定义的元类,可以对类的创建进行拦截。可以对类名、继承的父类、属性(方法)做一些预处理。
例如:将类名大写,默认继承object类,添加、修改属性(方法)名(私有属性的伪私有化就是通过修改属性名实现的)。
装饰器是对函数进行功能扩展(不用修改原代码),而元类可以对类进行功能扩展(添加额外的属性/方法)。
demo.py(用函数指定__metaclass__属性):
#-*- coding:utf-8 -*-def upper_attr(class_name, class_parents, class_attr): # class_name 会保存类的名字 Foo # class_parents 会保存类的父类 object # class_attr 会以字典的方式保存所有的类属性/方法 # 遍历属性字典,把不是__开头的属性名字变为大写 new_attr = {} for name, value in class_attr.items(): if not name.startswith("__"): new_attr[name.upper()] = value # 调用type来创建一个类 return type(class_name, class_parents, new_attr)class Foo(object, metaclass=upper_attr): # python3的方式 # python2.x的方式。 # __metaclass__ = upper_attr # 设置Foo类的元类为upper_attr bar = 'bip'print(hasattr(Foo, 'bar'))print(hasattr(Foo, 'BAR'))f = Foo()print(f.BAR)
demo.py(用类指定__metaclass__属性):
class UpperAttrMetaClass(type): # __new__ 是在__init__之前被调用的特殊方法 # __new__是用来创建对象并返回之的方法 # 而__init__只是用来将传入的参数初始化给对象 # 你很少用到__new__,除非你希望能够控制对象的创建 # 这里,创建的对象是类,我们希望能够自定义它,所以我们这里改写__new__ # 如果你希望的话,你也可以在__init__中做些事情 # 还有一些高级的用法会涉及到改写__call__特殊方法,但是我们这里不用 def __new__(cls, class_name, class_parents, class_attr): # 遍历属性字典,把不是__开头的属性名字变为大写 new_attr = {} for name, value in class_attr.items(): if not name.startswith("__"): new_attr[name.upper()] = value # 方法1:通过'type'来做类对象的创建 return type(class_name, class_parents, new_attr) # 方法2:复用type.__new__方法 # 这就是基本的OOP编程,没什么魔法 # return type.__new__(cls, class_name, class_parents, new_attr)# python3的用法class Foo(object, metaclass=UpperAttrMetaClass): bar = 'bip'# python2的用法# class Foo(object):# __metaclass__ = UpperAttrMetaClass# bar = 'bip'print(hasattr(Foo, 'bar'))# 输出: Falseprint(hasattr(Foo, 'BAR'))# 输出:Truef = Foo()print(f.BAR)# 输出:'bip'
更多关于Python相关内容感兴趣的读者可查看本站专题:《Python面向对象程序设计入门与进阶教程》、《Python数据结构与算法教程》、《Python函数使用技巧总结》、《Python字符串操作技巧汇总》、《Python编码操作技巧总结》及《Python入门与进阶经典教程》
希望本文所述对大家Python程序设计有所帮助。
122 在
学历:一种延缓就业设计,生活需求下的权衡之选中评论 工作几年后,报名考研了,到现在还没认真学习备考,迷茫中。作为一名北漂互联网打工人..123 在
Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..原梓番博客 在
在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..博主 在
佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..1111 在
佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..
Copyright·© 2019 侯体宗版权所有·
粤ICP备20027696号