举例讲解Python中metaclass元类的创建与使用


Posted in Python onJune 30, 2016

元类是可以让你定义某些类是如何被创建的。从根本上说,赋予你如何创建类的控制权。
元类也是一个类,是一个type类。
 
元类一般用于创建类。在执行类定义时,解释器必须要知道这个类的正确的元类,如果此属性没有定义,它会向上查找父类中的__metaclass__属性。如果还没发现,就查找全局变量。
 
对于传统类来说,它们的元类是types.ClassType。
 
元类也有构造器,传递三个参数:类名,从基类继承数据的元组,和类属性字典。
下面我们来定义一个元类,要求写类的时候必须给类提供一个__str__()方法,如果没有提供__repr__()方法,
则给你警告。

from warnings import warn
#元类需要继承type类
class ReqStrSugRepr(type):
  def __init__(cls, name, bases, attrd):
  #构造函数需要传递的参数为类名,基类,类属性字典
    super(ReqStrSugRepr, cls).__init__(name, bases, attrd)
    # 判断__str__字符串是否在类的属性字典里
    if '__str__' not in attrd:
      raise TypeError('Class requires overriding of __str__()')

    if '__repr__' not in attrd:
      warn('Class suggests overriding of __repr__()\n', stacklevel=3)

class Foo(object):
  #给类指定元类 
  __metaclass__ = ReqStrSugRepr

  def foo(self):
    pass
#这一段代码不用创建类来测试,直接运行一下就会报错,可见元类的功力。

举例讲解Python中metaclass元类的创建与使用

type

type函数可以查看一个变量的类型, 比如:

# <type 'int'>
# <type 'str'>
type(1)     
type('mink')

type函数还可以创建一个新的对象
type接受三个参数,name, bases, dict 第一个接受类名,第二个参数接受父类(元组形式),第三个参数接受属性和方法(字典形式)

X = type('X', (object,), dict(a=1))
# 等于
class X(object):
  a = 1

下面是接受函数的方法

def say(self):
  print 'hello'

X = type('X', (object,), dict(say=say))
x = X()

# pirnt hello
x.say()

元类

我们都知道通过类可以创建处实例对象,而元类就是创建出类对象的类。type可以创建出类对象也就是说type就是一个元类。

metaclass 属性

如果想使用元类创建类对象就需要对该对象添加一个__metaclass__属性。当然你首先得有一个元类

class PrivateMetaclass(type):
  def __new__(cls, name, parents, attrs):
    attrs = dict(('__%s' % k, v) for k, v in attrs.itmes())
    return super(PrivateMetaclass, cls).__new__(cls, name, parents, attrs)

class A(object):
  __metaclass__ = PrivateMetaclass
  a = 1
  b = 2

a = A()
# raise AttributeError
print a.a, a.b 

# print 1, 2
print a.__a, a.__b

这样你就可以通过元类来修改类的一些特性,上面的就是修改变量为私有变量.

Python 相关文章推荐
Python实现的多线程端口扫描工具分享
Jan 21 Python
利用Python绘制数据的瀑布图的教程
Apr 07 Python
Python3通过Luhn算法快速验证信用卡卡号的方法
May 14 Python
Python opencv实现人眼/人脸识别以及实时打码处理
Apr 29 Python
postman传递当前时间戳实例详解
Sep 14 Python
Python 实现大整数乘法算法的示例代码
Sep 17 Python
Python udp网络程序实现发送、接收数据功能示例
Dec 09 Python
Python发送邮件封装实现过程详解
May 09 Python
python爬取2021猫眼票房字体加密实例
Feb 19 Python
pandas:get_dummies()与pd.factorize()的用法及区别说明
May 21 Python
pandas DataFrame.shift()函数的具体使用
May 24 Python
opencv 分类白天与夜景视频的方法
Jun 05 Python
在Python中定义和使用抽象类的方法
Jun 30 #Python
Python中functools模块的常用函数解析
Jun 30 #Python
深入浅析Python中join 和 split详解(推荐)
Jun 30 #Python
Python列出一个文件夹及其子目录的所有文件
Jun 30 #Python
django之常用命令详解
Jun 30 #Python
全面了解Python环境配置及项目建立
Jun 30 #Python
浅谈Python 集合(set)类型的操作——并交差
Jun 30 #Python
You might like
php全排列递归算法代码
2012/10/09 PHP
yii2高级应用之自定义组件实现全局使用图片上传功能的方法
2016/10/08 PHP
PHP simplexml_load_file()函数讲解
2019/02/03 PHP
PHP安装扩展mcrypt以及相关依赖项深入讲解
2021/03/04 PHP
jquery 常用操作方法
2010/01/28 Javascript
js网页版计算器的简单实现
2013/07/02 Javascript
JS的get和set使用示例
2014/02/20 Javascript
js面向对象编程总结
2017/02/16 Javascript
深入理解vue中的$set
2017/06/01 Javascript
解决使用vue.js路由后失效的问题
2018/03/17 Javascript
Vue中控制v-for循环次数的实现方法
2018/09/26 Javascript
vuejs简单验证码功能完整示例
2019/01/08 Javascript
koa-router路由参数和前端路由的结合详解
2019/05/19 Javascript
纯异步nodejs文件夹(目录)复制功能
2019/09/03 NodeJs
用Golang运行JavaScript的实现示例
2019/11/25 Javascript
Vue自定义全局弹窗组件操作
2020/08/11 Javascript
vue 导航守卫和axios拦截器有哪些区别
2020/12/19 Vue.js
[13:18]《一刀刀一天》之DOTA全时刻21:详解TI新赛制 A队再露獠牙
2014/06/24 DOTA
python网络编程学习笔记(八):XML生成与解析(DOM、ElementTree)
2014/06/09 Python
python实现将元祖转换成数组的方法
2015/05/04 Python
Python实现批量检测HTTP服务的状态
2016/10/27 Python
详解python异步编程之asyncio(百万并发)
2018/07/07 Python
Python实现的ftp服务器功能详解【附源码下载】
2019/06/26 Python
Python3内置模块之base64编解码方法详解
2019/07/13 Python
python实现知乎高颜值图片爬取
2019/08/12 Python
wxPython实现列表增删改查功能
2019/11/19 Python
Python 面向对象之类class和对象基本用法示例
2020/02/02 Python
python批量提取图片信息并保存的实现
2021/02/05 Python
数据库专业英语
2012/11/30 面试题
运动会领导邀请函
2014/02/05 职场文书
经典的毕业生自荐信范文
2014/04/14 职场文书
大学英语演讲稿范文
2014/04/24 职场文书
乡镇干部个人对照检查材料思想汇报(原创篇)
2014/09/28 职场文书
丧事主持词
2015/07/02 职场文书
2016年母亲节广告语
2016/01/28 职场文书
golang 实用库gotable的具体使用
2021/07/01 Golang