一,理解类也是对象

在python中类同样也是一种对象,只要使用关键字class,Python解释器在执行的时候就会创建一个对象,这个对象(类)自身拥有创建对象(类实例)的能力,这就是为什么他是一个类的原因,但是,他的本质任然是一个对象。

class objectCreator(object):
pass my_object = objectCreator()
print(my_object) # 可以打印一个类,因为他其实就是一个对象 def echo(obj): # 可以将类作为参数传给函数
print(obj) echo(objectCreator) objectCreator.new_attribute = "foo" # 可以增加属性
print(hasattr(objectCreator, "new_attribute")) objectCreatorMirror = objectCreator # 可以赋值给你个变量
print(objectCreatorMirror())

二,动态地创建类

1,通过return class动态的构建需要的类

因为类也是对象,你可以在运行时动态的创建他们,就像他们任何对象一样,首先,你可以在函数中创建类,使用class关键字即可。

def choose_class(name):
if name == "foo":
class Foo(object):
pass return Foo
else:
class Bar(object):
pass return Bar MyClass = choose_class("foo")
print(MyClass) # 返回是类,不是类的实例 print(MyClass()) # 类的创建

2,通过type函数构造类,重新认识不一样的type

type可以动态创建类。type可以接受一个类的描述作为参数, 然后返回一个类。

type的语法:

type(类名,父类的元祖(针对继承的情况,可以为空), 包含属性的字典(名称和值))

如下代码:

class MyShinClass(object):
pass

可以手动通过type创建,其实

MyShinyClass = type("MyShinyClass", (), {})  # 返回一个类对象
print(MyShinyClass)
print(MyShinClass())

type创建类的范例:

1, 构建Foo类
class Foo(object):
bar = True Foo = type("Foo", (), {"bar": True}) 2, 继承Foo类
class FooChild(Foo):
pass FooChild = type("FooChild", (Foo,), {}) # 3, 为FooChild类增加方法
def echo_bar(self):
print(self.bar) FooChild = type("FooChild", (Foo,), {"echo_bar": echo_bar})
print(hasattr(FooChild, "echo_bar")) my_foo = FooChild()
my_foo.echo_bar()

可以看到,在python中,类也是对象,你可以动态的创建类。这就是当我们使用关键字class时python在幕后做的事情,而这就是通过元类实现的

三,元类

1,什么时元类

通过上文的描述,可以得到python中的类也是对象,元类就是用来创建这些类(对象)的。元类就是类的类。

函数type实际上就是一个元类,type就是python在背后用来创建所有类的元类。

元类就是创建类这种对象的东西。type就是python的内建元类,当然了,你也可以创建自己的元类

2, __metaclass__属性

可以在写一个类的时候为其添加__metaclass__属性,定义了__metaclass__就定义了这个类的元类

class Foo(object):
__metaclass__ = something class Foo(metaclass=something):
__metaclass__ = something

例如:如下代码

class Foo(Bar):
pass

在该类并定义的时候,他还没有在内存中生成,直到他被调用,Python做了如下的操作:

1)Foo中有__mataclass__这个属性吗?如果有,python会在内存中通过__metaclass__创建一个名字为Foo的类对象

2)如果Python没有找到__metaclass__,他就会在父类中寻找__metaclass__属性,并尝试做同样的操作。

3)如果Python没有找到__metaclass__,他就会在模块层中寻找__metaclass__,并尝试做同样的操作。

4)如果还是找不到__metaclass__,Python就会用内置的type来创建这个类对象

三,自定义元类

元类的主要目的就是当创建类的时候能自动地改变类,通常,你会为API做这样的事情, 你希望可以创建符合上下文的类。假象一个你的模块里所有累的属性都应该是大写形式,有好几种方法可以办到,但其中一种就是通过设定__metaclass__。采用这种方法,这个模块中所有的类都会通过这个元类创建。我们只需要告诉元类吧所有的属性都改成大写形式就万事大吉

class UpperAttrMetaClass(type):
def __new__(cls, name, bases, dic):
# 获取传入的属性值
attrs = ((name, value) for name, value in dic.items() if not name.startswith("__"))
# 将属性值变成大写,转换成字典
uppercase_attr = dict((name.upper(), value) for name, value in attrs) return type.__new__(cls, name, bases, uppercase_attr) # python3之后必须这样写
class Foo(metaclass=UpperAttrMetaClass):
bar = "bip" f = Foo()
print(hasattr(Foo, 'bar')) # False
print(hasattr(Foo, 'BAR')) # True

2,使用class来当作元类

由于__metaclass__必须返回一个类

四,使用__new__方法和元类方式分别实现单例模式

class Singleton(type):
def __init__(self, *args, **kwargs):
print("__init__")
self._instance = None
super(Singleton, self).__init__(*args, **kwargs) def __call__(self, *args, **kwargs):
print("__call__")
if self._instance is None:
self._instance = super(Singleton, self).__call__(*args, **kwargs)
return self._instance class Foo(metaclass=Singleton):
pass foo1 = Foo()
foo2 = Foo() print(Foo.__dict__)
print(foo1 is foo2) # __init__
# __call__
# __call__ # {'__module__': '__main__', '__dict__': <attribute '__dict__' of 'Foo' objects>,
# '__weakref__': <attribute '__weakref__' of 'Foo' objects>, '__doc__': None,
# '_instance': <__main__.Foo object at 0x102a034e0>} # True

基于这个例子:

我们知道元类生成的实例是一个类,而这里我们仅仅需要对这个实例增加一个属性来判断和保存生成的单例。

关于__call__方法的调用,因为Foo是Singleton的一个实例,所以Foo()这样的方式就调用了Singleton的__call__方法。

实现:

定义一个元类方法,实现函数方法的扩展自动识别,加入类方中中

class MetaClassTest(type):
def __new__(cls, name, bases, attrs):
count = 0
attrs["__FuncTest__"] = []
for k, v in attrs.items():
if "Chen_" in k:
attrs["__FuncTest__"].append(k)
count += 1
attrs["__FuncCount__"] = count return type.__new__(cls, name, bases, attrs) class MainTest(object, metaclass=MetaClassTest): def get_data(self, callback):
if hasattr(self, callback):
result = getattr(self, callback)()
print(result, "执行完毕~~~") # 以后只要每次扩展Chen_开头的函数,就会自动执行。
def Chen_Run(self):
return "Chenrun" def Chen_Peng(self):
return "ChenPeng" def run():
obj = MainTest()
for func in range(obj.__FuncCount__):
obj.get_data(callback=obj.__FuncTest__[func]) if __name__ == '__main__':
run()

理解python中的元类的更多相关文章

  1. [转]深刻理解Python中的元类(metaclass)以及元类实现单例模式

    使用元类 深刻理解Python中的元类(metaclass)以及元类实现单例模式 在看一些框架源代码的过程中碰到很多元类的实例,看起来很吃力很晦涩:在看python cookbook中关于元类创建单例 ...

  2. 深刻理解Python中的元类metaclass(转)

    本文由 伯乐在线 - bigship 翻译 英文出处:stackoverflow 译文:http://blog.jobbole.com/21351/ 译注:这是一篇在Stack overflow上很热 ...

  3. 深刻理解Python中的元类(metaclass)

    译注:这是一篇在Stack overflow上很热的帖子.提问者自称已经掌握了有关Python OOP编程中的各种概念,但始终觉得元类(metaclass)难以理解.他知道这肯定和自省有关,但仍然觉得 ...

  4. [转] 深刻理解Python中的元类(metaclass)

    非常详细的一篇深入讲解Python中metaclass的文章,感谢伯乐在线-bigship翻译及作者,转载收藏. 本文由 伯乐在线 - bigship 翻译.未经许可,禁止转载!英文出处:stacko ...

  5. 深刻理解Python中的元类(metaclass)【转】

    译注:这是一篇在Stack overflow上很热的帖子.提问者自称已经掌握了有关Python OOP编程中的各种概念,但始终觉得元类(metaclass)难以理解.他知道这肯定和自省有关,但仍然觉得 ...

  6. 深刻理解Python中的元类(metaclass)以及元类实现单例模式

    在理解元类之前,你需要先掌握Python中的类.Python中类的概念借鉴于Smalltalk,这显得有些奇特.在大多数编程语言中,类就是一组用来描述如何生成一个对象的代码段.在Python中这一点仍 ...

  7. 深入理解Python中的元类(metaclass)

    原文 译注:这是一篇在Stack overflow上很热的帖子.提问者自称已经掌握了有关Python OOP编程中的各种概念,但始终觉得元类(metaclass)难以理解.他知道这肯定和自省有关,但仍 ...

  8. 深度理解python中的元类

    本文转自:(英文版)https://stackoverflow.com/questions/100003/what-is-a-metaclass-in-python   (翻译版)   http:// ...

  9. python——深刻理解Python中的元类(metaclass)

    译注:这是一篇在Stack overflow上 很热的帖子.提问者自称已经掌握了有关Python OOP编程中的各种概念,但始终觉得元类(metaclass)难以理解.他知道这肯定和自省有关,但仍然觉 ...

  10. 理解 Python 中的元类

    本文编程环境:Jupyter NoteBook python3 类也是对象 在大多数编程语言中,类就是一组用来描述如何生成一个对象的代码段.在 Python 中这一点仍然成立: class Objec ...

随机推荐

  1. LG1343 地震逃生

    题意 汶川地震发生时,四川**中学正在上课,一看地震发生,老师们立刻带领x名学生逃跑,整个学校可以抽象地看成一个有向图,图中有n个点,m条边.1号点为教室,n号点为安全地带,每条边都只能容纳一定量的学 ...

  2. hadoop之 distcp(分布式拷贝)

    概述 distcp(分布式拷贝)是用于大规模集群内部和集群之间拷贝的工具. 它使用Map/Reduce实现文件分发,错误处理和恢复,以及报告生成. 它把文件和目录的列表作为map任务的输入,每个任务会 ...

  3. 2.JMeter查看结果树返回编码格式Unicode转为中文方法

    在使用JMeter做接口测试时,发现相同url,用postman工具,其返回数据参数为中文,而用JMeter工具,其返回参数为Unicode,如下图所示 解决方法如下: 1.Jmeter在对应的请求上 ...

  4. OpenSSL生成公钥私钥***

    证书标准 X.509 - 这是一种证书标准,主要定义了证书中应该包含哪些内容.其详情可以参考RFC5280,SSL使用的就是这种证书标准. 编码格式 同样的X.509证书,可能有不同的编码格式,目前有 ...

  5. Java 成员变量和局部变量

    1.成员变量 在类中定义,用来描述对象将要有什么. 2.局部变量 在类的方法中定义,在方法中临时保存数据. 成员变量和局部变量的区别 作用域不同: 局部变量的作用域仅限于定义它的方法 成员变量的作用域 ...

  6. 1094 The Largest Generation

    题意:略. 思路:层序遍历:在结点中增加一个数据域表示结点所在的层次. 代码: #include <cstdio> #include <queue> #include < ...

  7. oozie工作流相关入门整理

        Oozie支持工作流,其定义通过将多个Hadoop Job的定义按照一定的顺序组织起来,然后作为一个整体按照既定的路径运行.一个工作流已经定义了,通过启动该工作流Job,就会执行该工作流中包含 ...

  8. MySQL升级指南

    一 .MySQL升级 1.官方升级策略 注意 升级过程中必须使用具有管理权限的MySQL帐户来执行SQL语句. 1.升级方法 逻辑升级: 涉及使用 mysqldump从旧的MySQL版本导出现有数据 ...

  9. 浅谈PHP面向对象编程(九、设计模式)

    9.0 设计模式 在编写程序时经常会遇到一此典型的问题或需要完成某种特定需求,设计模式就是针对这些问题和需求,在大量的实践中总结和理论化之后优选的代码结构编程风格,以及解决问题的思考方式. 设计模式就 ...

  10. 第一章 : Android Studio 介绍 [Learn Android Studio 汉化教程]

    摘自:http://ask.android-studio.org/?/question/789,为便于学习重新整理.. 本章将引导您完成安装和设置开发环境,然后你就可以跟随本书的例子和课程学习. 首先 ...