面向对象高级B(元类)
元类
python一切皆对象,类实际上也是一个一个对象
类是一个对象,那他一定是由一个类实例化得到,这个类就叫元类
如何找元类
class Person:
def __init__(self, name):
self.name = name
def score(self):
print('分数是100')
print(type(Person))
#<class 'type'>
#所有类的元类都是type
常规的class 类名 会把类构造出来,实际上是元类实例化产生类这个对象,
Person类是对象,一定是由一个类实例化产生的,所以type()产生对象,调用__init__方法,看源码里面需要穿三个参数
**class底层实现原理是: type(object_or_name, bases, dict), **
object_or_name: 类的名字,是个字符串
bases: 是它的所有父类,基类
dict: 名称空间是一个字典
#通过type来直接产生类,不用class关键字
l = {}
exec{'''
school='oldboy'
def __init__(self, name):
self.name = name
def score(self):
print('分数是100')
''', {}, l}
def __init__(self, name):
self.name=name
Person = type('Person', (object,), l) # 等同于生成一个Person类,其中具有school属性,对象有name属性,有score方法
### 下面几句可以生成一个Person类,与上面的语句实现相同的功能
#def __init__(self, name):
#self.name = name
#Person = type('Person', (object,), {'school':'
# oldboy','__init__':__init__})
自定义元类
来控制类的产生,可以控制类名,可以控制类的继承关系 ,控制类的名称空间
自定义元类必须继承type,写一个类继承type,这种类都叫元类
class Mymeta(type):
def __init__(self, name, bases,dic):
#练习一:加限制 控制类名必须以sb 开头
if not name.startswith('sb'):
raise Exception(’类名没有以sb开头‘)# 如果不是以sb开头,就会抛异常
#练习二:类必须加注释
print(self.__dict__['__doc__']) # 有注释的话会把注释打印出来
#metaclass=Mymeta 指定这个类生成的时候,用自己写的Mymeta这个元类
class Person(object,metaclass=Mymeta):
'''
注释
'''
school = 'oldboy'
def __init__(self, name):
self.name = name
def score(self):
print('分数是100')
p = Person('nick')
- 自定义一个元类,控制类产生,类的名称空间必须有name字段才能创建成功,否则失败
class Mymeta(type):
def __init__(self, x, y, z):
for k, v in self.__dict__.items():
print(k,v)
# print(self.__dict__)
if 'name' not in self.__dict__.values() and 'name'not in self.__dict__.keys():
raise Exception('你的类没有“name”这个字段')
else:
print('名称空间已经存在"name"')
- 自定义一个元类,定义一个类,继承字典,使其具备 点取值和赋值功能,通过元类控制对象的产生,把所有对象的属性都放在attr字典中,属性全部删除
例如di=Mydict(name='lqz', age=18) di的名称空间中没有name和age属性,但是有attr字典属性,字典包含{'name':'lqz', 'age':18}
(待修改)
class Mymeta(type):
def __call__(self, name, age):#要有返回值
obj = object.__new__(self)
obj.__dict__['attr'] = {'name':name,'age':age}
return obj
class Person(metaclass=Mymeta):
def __init__(self,name,age):
self.name = name
self.age = age
p = Person('lje', 18)
print(p.__dict__)
print(p.attr['name'])
'''
{'attr': {'name': 'lje', 'age': 18}}
lje
'''
##延伸(对类名称空间中的属性单独放到attr字典里,但是该字典依然在类名称空间的字典里)
class Mymeta(type):
def __init__(self,name,bases,dic):
pass
def __new__(cls, name,bases,dic):#要有返回值
dic2={'attr':{}}#单独列一个字典,用于盛放类的属性
for k,v in dic.items():
if not k.startswith('__'):#如果key值以__开头,就把大拿出来放到dic2中
dic2['attr'][k]=v
print('-------',dic2)
return type.__new__(cls,name,bases,dic2)#把dic2当成名称空间返回,但是类的名称空间还是原来的dic,只不过属性放到attr小字典里了
class Person(metaclass=Mymeta);#Person=Mymeta(name,bases,dic) 调用type的__call__,内部调用了Mymeta.__new__,又调用了Mymeta的__init__
school = 'Tshinghua'
age = 10
def __init__(self, name, age):
self.name = name
self.age = age
print(Person.__dict__)
print(Person.attr['school'])
'''
------- {'attr': {'school': 'oldboy', 'age': 10}}
{'attr': {'school': 'oldboy', 'age': 10}, '__module__': '__main__', '__dict__': <attribute '__dict__' of 'Person' objects>, '__weakref__': <attribute '__weakref__' of 'Person' objects>, '__doc__': None}
Tshinghua
'''
面向对象高级B(元类)的更多相关文章
- Python面向对象篇之元类,附Django Model核心原理
关于元类,我写过一篇,如果你只是了解元类,看下面这一篇就足够了. Python面向对象之类的方法和属性 本篇是深度解剖,如果你觉得元类用不到,呵呵,那是因为你不了解Django. 在Python中有一 ...
- python 面向对象进阶之元类metaclass
一:知识储备 exec exec:三个参数 参数一:字符串形式的命令 参数二:全局作用域(字典形式),如果不指定,默认为globals() 参数三:局部作用域(字典形式),如果不指定,默认为local ...
- C++面向对象高级编程(五)类与类之间的关系
技术在于交流.沟通,转载请注明出处并保持作品的完整性. 本节主要介绍一下类与类之间的关系,也就是面向对象编程先介绍两个术语 Object Oriented Programming OOP面向对象编 ...
- 【转】python面向对象中的元类
type() 动态语言和静态语言最大的不同,就是函数和类的定义,不是编译时定义的,而是运行时动态创建的. 比方说我们要定义一个Hello的class,就写一个hello.py模块: class Hel ...
- python 面向对象编程 之 元类
元类是类的类,使我们自定义的类,即我们用class定义类本质就是元类,是类的模板 四步走: 一:控制class定义类的过程 1.先拿到类名 2.在拿到基类 3.执行类体代码,得到名称空间的dict 4 ...
- python 面向对象十二 元类
一.类也是对象 只要使用关键字class,Python解释器在执行的时候就会创建一个对象.下面的代码段: class ObjectCreator(object): pass 将在内存中创建一个对象,名 ...
- CSIC_716_20191129【面向对象高级----反射、类的内置方法(魔法方法)、单例模式】
反射 反射是通过'字符串'对 对象的属性进行操作,反射有四个内置的方法. hasattr 通过字符串 判断对象的属性或者方法是否存在 getattr 通过字符串 获取对象的属性或者方法 ...
- 【IOS 开发】Objective - C 面向对象高级特性 - 包装类 | 类处理 | 类别 | 扩展 | 协议 | 委托 | 异常处理 | 反射
一. Objective-C 对象简单处理 1. 包装类 (1) 包装类简介 NSValue 和 NSNumber : -- 通用包装类 NSValue : NSValue 包装单个 short, i ...
- python面向对象高级:定制类
Python的class中还有许多这样有特殊用途的函数,可以帮助我们定制类. 比如: __str__ 与__repr____iter____getitem____call__ __str__ 与__r ...
- Python实用笔记 (26)面向对象高级编程——定制类
Python的class允许定义许多定制方法,可以让我们非常方便地生成特定的类.以下是集中常见的定制方法: 怎么才能打印得好看呢?只需要定义好__str__()方法,返回一个好看的字符串就可以了: _ ...
随机推荐
- java循环控制语句loop使用
java中break和continue可以跳出指定循环,break和continue之后不加任何循环名则默认跳出其所在的循环,在其后加指定循环名,则可以跳出该指定循环(指定循环一般为循环嵌套的外循环) ...
- vue 学习注意事项
一:插值方式: 1:数据绑定,最常见的形式就是使用 “Mustache” 语法(双大括号)的文本插值 <span>Message: {{ msg }}</span> 通过使用 ...
- 路径规划基础A*算法
1,Dijkstra’s 算法 一种发散性寻找最短路径算法. 由起点开始向四周开始发散,直到碰到目标点为止.这时就是最短路径.优点:能找到与目标点的最短路径:缺点:搜索花费的时间会比较长. 2,Gr ...
- luogu P1533 可怜的狗狗 |莫队+二分
题目背景 小卡由于公务需要出差,将新家中的狗狗们托付给朋友嘉嘉,但是嘉嘉是一个很懒的人,他才没那么多时间帮小卡喂狗狗. 题目描述 小卡家有N只狗,由于品种.年龄不同,每一只狗都有一个不同的漂亮值.漂亮 ...
- orientation属性(判断是否为横竖屏)
现在有一个需求:移动端网页默认竖屏显示,当用户横屏浏览,就给予相应提示,比如横屏时显示下面截图提示信息 几年前,可能大家想到用 window.orientation 属性来实现,现官方已弃用,不做推荐 ...
- 基础知识---IEnumerable、ICollection、IList、IQueryable
一.定义 IEnumerable public interface IEnumerable<out T> : IEnumerable ICollection public interfac ...
- idea更改之前保存的git账号和密码
1.打开控制面板 2.用户账户 3.管理windows凭据 4.点击里面的git就可以修改了
- NetCoreApi框架搭建三、AutoFac 依赖入注)
这里不多做理论上的解释,因为我感觉自己也不是很完全的理解,所以只是记录我自己做的过程. 首先还是粘贴大神的链接:https://www.cnblogs.com/RayWang/p/11165509.h ...
- 深入理解 Kubernetes 资源限制:CPU
原文地址:https://www.yangcs.net/posts/understanding-resource-limits-in-kubernetes-cpu-time/ 在关于 Kubernet ...
- android studio学习----构建(gradle )依赖时使用动态依赖的问题
今天在看Dan Lew大神的博客发现最新的文章就是 “Don't use dynamic versions for your dependencies” Everyone, please, to st ...