python面向对象之类中的内置方法
- __setattr__,__delattr__,__getattr__,__getattribute__以及标准类型的二次加工
- __setitem__,__getitem__,__delitem__
- __str__,__repr__,__format__
- __slots__
- __doc__
- __module__,__class__
- __del__
- __call__
- __iter__,__next__(迭代器协议)
- __get__,__set__,__delete__(描述符)
- __enter__,__exit__(with方法和上下文管理协议)
- __new__
__setattr__,__delattr__,__getattr__,__getattribute__以及标准类型的二次加工
__setattr__,__delattr__,__getattr__的用法
class Foo: x=1 def __init__(self,y): self.y=y def __getattr__(self, item): #__getattr__只有在使用点调用属性且属性不存在的时候才会触发 print('----> from getattr:你找的属性不存在') def __setattr__(self, key, value): # __setattr__添加/修改属性会触发它的执行 将key的值改为value print('----> from setattr') print({key:value}) self.__dict__[key] = value def __delattr__(self, item): #除属性的时候会触发__delattr__ item=删除的属性值 print('----> from delattr--%s'%item) self.__dict__.pop(item) f1=Foo('a') #触发了__setattr__ #因为重写了__setattr__,凡是赋值操作都会触发它的运行,所以根本没赋值,除非直接操作属性字典,否则永远无法赋值 print(f1.__dict__) f1.z=3 print(f1.__dict__) #添加属性也不可行 f1.xxx #_使用点调用属性且属性不存在的时候会触发 _getattr__ print(f1.__dict__) del f1.y #删除属性的时候会触发__delattr__ print(f1.__dict__)
二次加工标准类型(包装和授权)
包装
python提供了标准数据类型,以及丰富的内置方法,其实在很多场景下我们都需要基于标准数据类型来定制我们自己的数据类型,新增/改写方法,这就用到了继承/派生知识(其他的标准类型均可以通过下面的方式进行二次加工)
class List(list): def __init__(self,item,tag=False): super().__init__(item) self.tag=tag def append(self, object): #派生自己的append,加上判断 if not isinstance(object,str): raise TypeError('must be str') super().append(object) @property def get_mid(self): #增加自己的取中间数方法 index=len(self)//2 return self[index] def clear(self): #派生自己的clear方法,加上限制 if not self.tag: raise TypeError('别乱删') super().clear() l1=List('hantao') print(l1) l1.append('asdfd') print(l1) # l1.append(12) #抛异常 print(l1.get_mid) # l1.clear() # print(l1) l1.tag=True l1.clear()
授权
授权是包装的一个特性, 包装一个类型通常是对已存在的类型的一些定制,这种做法可以新建,修改或删除原有产品的功能。其它的则保持原样。授权的过程,即是所有更新的功能都是由新类的某部分来处理,但已存在的功能就授权给对象的默认属性。
实现授权的关键点就是覆盖__getattr__方法
import time class MyOpen(object): def __init__(self,file_path,MODE='r',encoding='utf-8'): self.file=open(file_path,MODE,encoding=encoding) def write(self,line): #加入新的功能 t=time.strftime('%Y-%m-%d %T') self.file.write('%s%s'%(t,line)) def __getattr__(self, item): #默认功能授权给对象的默认属性 if hasattr(self.file,item): return getattr(self.file,item) raise TypeError('无此方法') myopen=MyOpen('a.txt','w+') print(myopen.read) myopen.write('ojbk\n') myopen.write('jbok\n') myopen.write('ijkb\n')
__getattribute__
class Foo: def __init__(self,x): self.x=x def __getattr__(self, item): print('执行的是我') # return self.__dict__[item] def __getattribute__(self, item): print('不管是否存在,我都会执行') raise AttributeError('哈哈') f1=Foo(10) f1.x f1.xxxxxx #当__getattribute__与__getattr__同时存在,只会执行__getattrbute__,除非__getattribute__在执行过程中抛出异常AttributeError 二者同时出现
__setitem__,__getitem__,__delitem__
class Foo: def __init__(self,x): self.x=x def __setitem__(self, key, value): #以字典的形式改变或添加属性时触发 print('执行了setattr') self.__dict__[key]=value def __getitem__(self, item): #以字典的形式调用属性时触发 print('执行了getattr') return self.__dict__[item] def __delitem__(self, item): #以字典的形式删除属性时触发 print('执行了delattr') self.__dict__.pop(item) foo=Foo('han') foo['age']=100 #触发__setitem__ foo['x']='tao' #触发__setitem__ print(foo.__dict__) print(foo['x']) #触发__getattr__ del foo['x'] #触发__delattr__
__str__,__repr__,__format__
__str__,__repr__
改变对象的字符串显示
class Persion: def __init__(self,name,age): self.name=name self.age=age def __str__(self): #访问实例化对象时返回的字符串 return '%s--str'%self.name def __repr__(self): return '%s--repr'%self.name hantao=Persion('hantao',18) ''' str函数或者print函数--->obj.__str__() repr或者交互式解释器--->obj.__repr__() 如果__str__没有被定义,那么就会使用__repr__来代替输出 注意:这俩方法的返回值必须是字符串,否则抛出异常 ''' print(hantao) print('来自str',str(hantao)) print('来自repr',repr(hantao))
__format__
自定制格式化字符串
date_dic={ 'ymd':'{0.year}:{0.month}:{0.day}', 'dmy':'{0.day}/{0.month}/{0.year}', 'mdy':'{0.month}-{0.day}-{0.year}', } class Date: def __init__(self,year,month,day): self.year=year self.month=month self.day=day def __format__(self, format_spec): if not format_spec or format_spec not in date_dic: format_spec='ymd' fmt=date_dic[format_spec] return fmt.format(self) d1=Date(2016,12,29) print(format(d1)) print('{:mdy}'.format(d1))
__slots__
1.__slots__是什么:是一个类变量,变量值可以是列表,元祖,或者可迭代对象,也可以是一个字符串(意味着所有实例只有一个数据属性)
2.引子:使用点来访问属性本质就是在访问类或者对象的__dict__属性字典(类的字典是共享的,而每个实例的是独立的)
3.为何使用__slots__:字典会占用大量内存,如果你有一个属性很少的类,但是有很多实例,为了节省内存可以使用__slots__取代实例的__dict__.
4.如何实现:当定义__slots__后,__slots__就会为实例使用一种更加紧凑的内部表示。实例通过一个很小的固定大小的数组来构建,而不是为每个实例定义一个字典,这跟元组或列表很类似。在__slots__中列出的属性名在内部被映射到这个数组的指定小标上。使用__slots__一个不好的地方就是我们不能再给
实例添加新的属性了,只能使用在__slots__中定义的那些属性名。
5.注意事项:__slots__的很多特性都依赖于普通的基于字典的实现。另外,定义了__slots__后的类不再 支持一些普通类特性了,比如多继承。大多数情况下,你应该只在那些经常被使用到 的用作数据结构的类上定义__slots__比如在程序中需要创建某个类的几百万个实例对象 。
6.关于__slots__的一个常见误区是它可以作为一个封装工具来防止用户给实例增加新的属性。尽管使用__slots__可以达到这样的目的,但是这个并不是它的初衷。更多的是用来作为一个内存优化工具。
class Foo: __slots__=['name','age'] f1=Foo() f1.name='alex' f1.age=18 print(f1.__slots__) f2=Foo() f2.name='egon' f2.age=19 print(f2.__slots__) print(Foo.__dict__) #f1与f2都没有属性字典__dict__了,统一归__slots__管,节省内
__doc__
描述类的描述信息,无法被继承
__module__,__class__
__module__ 表示当前操作的对象在那个模块
__class__表示当前操作的对象的类是什么
在test模块中建一个类:
class Foo: def __init__(self,x): self.x=x
调用test模块中的类:
from test import Foo foo=Foo('h') print(foo.__module__) print(foo.__class__)
结果:
__del__
析构方法,当对象在内存中被释放时,自动触发执行。
注:如果产生的对象仅仅只是python程序级别的(用户级),那么无需定义__del__,如果产生的对象的同时还会向操作系统发起系统调用,即一个对象有用户级与内核级两种资源,比如(打开一个文件,创建一个数据库链接),则必须在清除对象的同时回收系统资源,这就用到了__del__
class Foo: def __del__(self): print('执行我啦') f1=Foo() del f1 print('------->') #输出结果 执行我啦 ------->
典型的应用场景:
创建数据库类,用该类实例化出数据库链接对象,对象本身是存放于用户空间内存中,而链接则是由操作系统管理的,存放于内核空间内存中
当程序结束时,python只会回收自己的内存空间,即用户态内存,而操作系统的资源则没有被回收,这就需要我们定制__del__,在对象被删除前向操作系统发起关闭数据库链接的系统调用,回收资源
这与文件处理是一个道理
f=open('a.txt') #做了两件事,在用户空间拿到一个f变量,在操作系统内核空间打开一个文件
del f #只回收用户空间的f,操作系统的文件还处于打开状态
#所以我们应该在del f之前保证f.close()执行,即便是没有del,程序执行完毕也会自动del清理资源,于是文件操作的正确用法应该是
f=open('a.txt')
读写...
f.close()
很多情况下大家都容易忽略f.close,这就用到了with上下文管理
__call__
对象后面加括号,触发执行。
注:构造方法的执行是由创建对象触发的,即:对象 = 类名() ;而对于 __call__ 方法的执行是由对象后加括号触发的,即:对象() 或者 类()()
class Foo: def __init__(self): pass def __call__(self, *args, **kwargs): print('__call__') obj = Foo() # 执行 __init__ obj() # 执行 __call__
__iter__,__next__(迭代器协议)
实现迭代器协议
class Foo: def __init__(self,x,stop): self.x=x self.stop=stop def __iter__(self): return self def __next__(self): if self.x >self.stop: raise StopIteration self.x+=1 return self.x f=Foo(5,10) print(next(f))
class Fib: def __init__(self): self._a=0 self._b=1 def __iter__(self): return self def __next__(self): self._a,self._b=self._b,self._a + self._b return self._a f1=Fib() print(f1.__next__()) print(next(f1)) print(next(f1)) for i in f1: if i > 100: break print('%s ' %i,end='')
__get__,__set__,__delete__(描述符)
描述符的定义:
描述符本质是一个新式类,在这个新式类中,至少实现了__get__(),__set__(),__delete__()中的一个,这也被称为描述符协议
class Foo: def __get__(self, instance, owner): print('--get方法') def __set__(self, instance, value): print('--set方法') def __delete__(self, instance): print('--delete方法')
各方法的作用:
__get__():调用一个属性时,触发
__set__():为一个属性赋值时,触发
__delete__():采用del删除属性时,触发
描述符的作用是用来代理另外一个类的属性的(必须把描述符定义成这个类的类属性,不能定义到构造函数中)
class Foo: def __get__(self, instance, owner): print('--get方法') def __set__(self, instance, value): print('--set方法') def __delete__(self, instance): print('--delete方法') class Test: x=Foo() def __init__(self,n): self.x=n f=Test('a') #此时会调用描述符
描述符的分类:
数据描述符:至少实现了__get__()和__set__()
非数据描述符:没有实现__set__()
注意事项:
一 描述符本身应该定义成新式类,被代理的类也应该是新式类
二 必须把描述符定义成这个类的类属性,不能为定义到构造函数中
调用优先级:
1.类属性
2.数据描述符
3.实例属性
4.非数据描述符
5.找不到的属性触发__getattr__()
通过描述符机制来实现类型限制功能
class Typed(): def __init__(self,name,expected_type): self.name=name self.expected_type=expected_type def __set__(self, instance, value): print('set') if not isinstance(value,self.expected_type): raise TypeError('类型错误') instance.__dict__[self.name]=value def __get__(self, instance, owner): print('get') if instance is None: return self return instance.__dict__[self.name] def __delete__(self, instance): print('delete') instance.__dict__.pop(self.name) class People: name=Typed('name',str) age=Typed('age',int) hobbies=Typed('hobbies',list) def __init__(self,name,age,hobbies): self.name=name self.age=age self.hobbies=hobbies
class Typed: def __init__(self,name,expected_type): self.name=name self.expected_type=expected_type def __get__(self, instance, owner): print('get--->',instance,owner) if instance is None: return self return instance.__dict__[self.name] def __set__(self, instance, value): print('set--->',instance,value) if not isinstance(value,self.expected_type): raise TypeError('Expected %s' %str(self.expected_type)) instance.__dict__[self.name]=value def __delete__(self, instance): print('delete--->',instance) instance.__dict__.pop(self.name) def typeassert(**kwargs): def decorate(cls): print('类的装饰器开始运行啦------>',kwargs) for name,expected_type in kwargs.items(): setattr(cls,name,Typed(name,expected_type)) return cls return decorate @typeassert(name=str,age=int,salary=float) #有参:1.运行typeassert(...)返回结果是decorate,此时参数都传给kwargs 2.People=decorate(People) class People: def __init__(self,name,age,salary): self.name=name self.age=age self.salary=salary print(People.__dict__) p1=People('egon',18,3333.3)
__enter__,__exit__(with方法和上下文管理协议)
以打来文件举例:
f=open('a.txt') f.read() f.close() || || || \/ with open('a.txt') as f: f.read()
上下文管理协议就是利用with语句,无需自己关闭以及打开的对象
with语句的实际是通过类中的__enter__,和__exit__实现的
class MyOpen: def __init__(self,name): self.name=name def __enter__(self): #with语句直接回执行enter方法,并将返回值赋值给as后边的变量 print('--enter方法') return self def __exit__(self, exc_type, exc_val, exc_tb): #再with语句执行完毕前,触发exit方法 print('--exit方法') with MyOpen('a.txt')as f: print('开始执行')
需要注意的是:
1.__exit__()中的三个参数分别代表异常类型,异常值和追溯信息,with语句中代码块出现异常,则with后的代码都无法执行
2.如果exit返回值是True,with语句应错误终止时不会终止总程序
执行过程:
1.with obj ----> 触发obj.__enter__(),并拿到返回值---->as f 赋值给f
2.开始执行with中的语句
---没有异常:运行完毕后执行__exit__()
---出现异常:直接触发__exit__(),三个值(异常类型,异常值和追溯信息)
------如果__exit__返回值不是True,报出异常,程序中断
------如果__exit__返回值是True,吞掉异常,结束with语句,其他程序继续执行
3.__exit__的运行完毕就代表着整个with语句的结束
用途或者说好处:
1.使用with语句的目的就是把代码块放入with中执行,with结束后,自动完成清理工作,无须手动干预
2.在需要管理一些资源比如文件,网络连接和锁的编程环境中,可以在__exit__中定制自动释放资源的机制,你无须再去关系这个问题,这将大有用处
__new__
class A: def __init__(self): self.x = 1 print('in init function') def __new__(cls, *args, **kwargs): print('in new function') return object.__new__(A, *args, **kwargs) a = A() print(a.x)