Python 使用元类type创建类对象常见应用详解


Posted in Python onOctober 17, 2019

本文实例讲述了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)
# 静态方法
@staticmethod
def print_static():
  print("----haha-----")
# 类方法
@classmethod
def 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()   # 100
b.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'))
# 输出: False
print(hasattr(Foo, 'BAR'))
# 输出:True
f = Foo()
print(f.BAR)
# 输出:'bip'

希望本文所述对大家Python程序设计有所帮助。

Python 相关文章推荐
Python实现的rsa加密算法详解
Jan 24 Python
TensorFlow用expand_dim()来增加维度的方法
Jul 26 Python
Python做智能家居温湿度报警系统
Sep 25 Python
对Python2与Python3中__bool__方法的差异详解
Nov 01 Python
Python爬虫 bilibili视频弹幕提取过程详解
Jul 31 Python
python 五子棋如何获得鼠标点击坐标
Nov 04 Python
pyhton中__pycache__文件夹的产生与作用详解
Nov 24 Python
python 导入数据及作图的实现
Dec 03 Python
pandas的相关系数与协方差实例
Dec 27 Python
python模拟预测一下新型冠状病毒肺炎的数据
Feb 01 Python
Python进程Multiprocessing模块原理解析
Feb 28 Python
python常量折叠基础知识点讲解
Feb 28 Python
Python with关键字,上下文管理器,@contextmanager文件操作示例
Oct 17 #Python
浅析Python+OpenCV使用摄像头追踪人脸面部血液变化实现脉搏评估
Oct 17 #Python
Python 3.8正式发布重要新功能一览
Oct 17 #Python
Python 装饰器@,对函数进行功能扩展操作示例【开闭原则】
Oct 17 #Python
python实现复制文件到指定目录
Oct 16 #Python
如何解决django-celery启动后迅速关闭
Oct 16 #Python
Python发送邮件的实例代码讲解
Oct 16 #Python
You might like
php正则表达式使用的详细介绍
2013/04/27 PHP
PHP邮件发送类PHPMailer用法实例详解
2014/09/22 PHP
PHP连接MSSQL时nvarchar字段长度被截断为255的解决方法
2014/12/25 PHP
thinkPHP简单导入和使用阿里云OSSsdk的方法
2017/03/15 PHP
常用js脚本
2006/12/03 Javascript
IE bug table元素的innerHTML
2010/01/11 Javascript
一段实现页面上的图片延时加载的js代码
2010/02/11 Javascript
JavaScript 浏览器验证代码(来自discuz)
2010/07/17 Javascript
为什么要在引入的css或者js文件后面加参数的详细讲解
2013/05/03 Javascript
javascript删除字符串最后一个字符
2014/01/14 Javascript
jquery easyui中treegrid用法的简单实例
2014/02/18 Javascript
在JavaScript中处理时间之setMinutes()方法的使用
2015/06/11 Javascript
Wireshark基本介绍和学习TCP三次握手
2016/08/15 Javascript
JavaScript中从setTimeout与setInterval到AJAX异步
2017/02/13 Javascript
通过学习bootstrop导航条学会修改bootstrop颜色基调
2017/06/11 Javascript
详解使用vue-cli脚手架初始化Vue项目下的项目结构
2018/03/08 Javascript
javascript数组常见操作方法实例总结【连接、添加、删除、去重、排序等】
2019/06/13 Javascript
python定时检查启动某个exe程序适合检测exe是否挂了
2013/01/21 Python
Python编程中归并排序算法的实现步骤详解
2016/05/04 Python
Python实现简易Web爬虫详解
2018/01/03 Python
pandas带有重复索引操作方法
2018/06/08 Python
对sklearn的使用之数据集的拆分与训练详解(python3.6)
2018/12/14 Python
itchat-python搭建微信机器人(附示例)
2019/06/11 Python
python实现二分类的卡方分箱示例
2019/11/22 Python
python给图像加上mask,并提取mask区域实例
2020/01/19 Python
使用py-spy解决scrapy卡死的问题方法
2020/09/29 Python
Python使用openpyxl复制整张sheet
2021/03/24 Python
个人承诺书
2014/03/26 职场文书
保健品市场营销方案
2014/03/31 职场文书
《孙权劝学》教学反思
2014/04/23 职场文书
优秀护士先进事迹
2014/05/08 职场文书
车辆委托书范本
2014/10/05 职场文书
作风建设剖析材料
2014/10/06 职场文书
出差报告范文
2014/11/06 职场文书
客服专员岗位职责范本
2015/04/07 职场文书
Linux安装Nginx步骤详解
2021/03/31 Servers