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实例一个类背后发生了什么
Feb 09 Python
Django 浅谈根据配置生成SQL语句的问题
May 29 Python
python微信公众号之关键词自动回复
Jun 15 Python
六行python代码的爱心曲线详解
May 17 Python
Python使用Pandas对csv文件进行数据处理的方法
Aug 01 Python
扩展Django admin的list_filter()可使用范围方法
Aug 21 Python
pygame实现非图片按钮效果
Oct 29 Python
浅谈对pytroch中torch.autograd.backward的思考
Dec 27 Python
python爬虫模拟浏览器访问-User-Agent过程解析
Dec 28 Python
Python PyQt5运行程序把输出信息展示到GUI图形界面上
Apr 27 Python
利用python控制Autocad:pyautocad方式
Jun 01 Python
写一个Python脚本自动爬取Bilibili小视频
Apr 24 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数组游标实现对数组的各种操作详解
2016/01/26 PHP
中高级PHP程序员应该掌握哪些技术?
2016/09/23 PHP
Laravel 连接(Join)示例
2019/10/16 PHP
关于PHP求解三数之和问题详析
2020/11/09 PHP
Javascript 获取字符串字节数的多种方法
2009/06/02 Javascript
javascript如何使用bind指定接收者
2014/05/04 Javascript
JavaScript代码编写中各种各样的坑和填坑方法
2014/06/06 Javascript
IE中图片的onload事件无效问题和解决方法
2014/06/06 Javascript
jQuery插件实现无缝滚动特效
2015/11/24 Javascript
jQuery的层级查找方式分析
2016/06/16 Javascript
详解从Node.js的child_process模块来学习父子进程之间的通信
2017/03/27 Javascript
你可能不知道的JSON.stringify()详解
2017/08/17 Javascript
vue history 模式打包部署在域名的二级目录的配置指南
2019/07/02 Javascript
jQuery实现鼠标滑动切换图片
2020/05/27 jQuery
vue-router 控制路由权限的实现
2020/09/24 Javascript
vue 数据遍历筛选 过滤 排序的应用操作
2020/11/17 Javascript
[36:45]TNC vs VGJ.S 2018国际邀请赛小组赛BO2 第二场 8.18
2018/08/19 DOTA
Python 忽略warning的输出方法
2018/10/18 Python
Python中的异常处理try/except/finally/raise用法分析
2019/02/28 Python
python简单鼠标自动点击某区域的实例
2019/06/25 Python
Python 用matplotlib画以时间日期为x轴的图像
2019/08/06 Python
python中time库的实例使用方法
2019/10/31 Python
python解析xml文件方式(解析、更新、写入)
2020/03/05 Python
Python实现曲线拟合的最小二乘法
2021/02/19 Python
AmazeUI在模态框中嵌入表单形成模态输入框
2020/08/20 HTML / CSS
爱游人:Travelliker
2017/09/05 全球购物
耐克奥地利官网:Nike奥地利
2019/08/16 全球购物
乌克兰网上珠宝商店:GoldSoveren
2020/03/31 全球购物
医院门卫岗位职责
2013/12/30 职场文书
中文师范生自荐信
2014/01/30 职场文书
汽车专业求职信
2014/06/05 职场文书
县政府领导班子“四风”方面突出问题整改措施
2014/09/23 职场文书
2014年建筑工程工作总结
2014/12/03 职场文书
入党申请书怎么写?
2019/06/11 职场文书
对Golang中的FORM相关字段理解
2021/05/02 Golang
MySQL数据库安装方法与图形化管理工具介绍
2022/05/30 MySQL