Python魔术方法专题


Posted in Python onJune 19, 2020

_del_

类的析构方法,它在对象被回收时执行,主要的作用时用来释放资源(内存 文件 进程等)

因为Python内存回收机制,使得Python的del方法的执行时间是不确定的,因此不推荐在Python中使用析构方法。

class Bar(object):
  def __del__(self):
    print("被回收了! ~")


a = Bar()

a.__del__() # 主动调用是没用的,因为引用计数不为零,并不会回收资源 gc
print("已经删除a了")
print(a)

del a
# print(a)

_dict_

  • 是一个绑定对象属性的字典 存储的是属性的 键值对应关系
  • 可以直接通过修改这个字典来为对象添加属性(但是不推荐这样做!会使得程序的可读性降低 破坏程序的结构 充分理解 后使用 但是也要慎重)甚至 你可以通过修改 dict 来为对象添加方法 例如 func

_slots_

  • 限定类的对象只能拥有某些属性,防止写错属性名,也可以实现不允许动态添加其他属性。
  • 形式:一个元组或 列表
  • 需要注意 一旦类指定了 slots 那就意味着 类的属性键值绑定关系 由__slots__来维护 也就是说 对象将没有 __dict__方法
  • __slots__只能约束本类,不能约束继承它的子类,如果子类也定义了slots 方法,那么对子类的约束将会成为两者的并集。
class Bar(object):
  __slots__ = ('name', 'gender')

  def __init__(self, name='monkey'):
    self.name = name
    self.gender = 'male'


a = Bar()
a.age = 18  # 动态添加属性是会报错的。
print(a.name)

_str_

必须返回一个str 类型 在打印对象的时候将会 打印返回的 str 而不是默认的 self.str
:return: <main.... object at 0x1084b7208>

class Bar(object):

  def __str__(self):
    return "Bar"


a = Bar()
print(a) # Bar

_repr_

将对象转化成对解释器友好的形式,它跟eval()方法联系紧密,通常repr()调用 对象的__repr__方法,该方法返回以字符串格式的 对解释器友好的 对象描述,eval() 可以将repr()的返回值 转化为原对象。

这玩意很强大,它是最直接的多态体现,几乎任何类对象都实现了它,但是每个返回的结果都是不一样的。

_class_

_class_ 允许通过对象调用类的方法和操作类的属性即 object.__class__ 可以拿到这个对象的类
拿到类后可以进行新的实例化 操作类的属性 调用类的方法等.

class Bar(object):
  name = 'monkey'


a = Bar()
print(a.__class__.name)  # 允许通过实例化对象访问类

_doc_

打印对象或类或方法的文档字符串

class Bar(object):
  """
  A simple show class!
  """
  name = 'monkey'

  def get_name(self):
    """
    get class argument name
    """
    return self.__class__.name


a = Bar()
print(a.__class__.__doc__)
print(a.__class__.get_name.__doc__)

#  A simple show class!
#  
#
#    get class argument name

_base_

用来返回类的父类

_bases_

用来返回类的继承列表

class Lady(object):
  """ """

class Small(object):
  """ """

class SmallLady(Small, Lady):
  """"""


print(Lady.__base__)  # <class 'object'>
print(SmallLady.__bases__)  # (<class '__main__.Small'>, <class '__main__.Lady'>)

_iter_

必须返回可迭代对象

这个对象需要实现__next__方法。

_next_

每次返回迭代器的下一个值或一个迭代异常来终止迭代。

_len_

每次返回迭代器的下一个值或一个迭代异常来终止迭代。

class ListMeta(type):
  def __call__(self, data, *args, **kwargs):  # 使得self 也就是实例化出的类 是可调用的 List() 这里的self指的是 将要 实例化出来的类 本身
    self.__init__(self,data)
    return self

  def __str__(self):
    result = self.clean_data(self) # 是 List 可以返回期望的列表格式 将对象转化为对人友好的字符串
    result = '[{}]'.format(result[:-1])
    return result

  def __repr__(self):
    return 'List({})'.format(self.__str__())  # 转化为对解释器友好的字符串

  def __iter__(self):		# 返回实现了迭代器协议的对象
    return self # 它本身实现了 __next__

  def __next__(self):		# 实现迭代器协议,每次返回下一个值 或 一个迭代异常终止迭代
    if self.index >= len(self.data):  
      raise StopIteration
    else:
      value = self.data[self.index]
      self.index += 1
      return value

  def __len__(self):  # 返回对象的长度,len()函数会执行对象的 __len__方法
    return self.len


class List(metaclass=ListMeta):

  def __init__(self, data):
    self.data = data
    self.index = 0
    self.len = len(self.data)
    

l = List([1,2,3,4,5,6,7])

print(l)
print(len(l))

for i in l:
  print(i)

_hash_

必须返回一个int类型的数据,并且可以唯一的表示这个对象。这点很重要。

_getattribute_

  • 此方法在每次访问对象的属性之前都会被调用,它容易使你陷入无限的递归中。
  • 如果需要对对象属性的访问做一些限制 譬如 以"block_" 开头的属性不允许访问可以这样来实现,这时候她是非常有用的。
  • 如果该方法找到了对象的属性,那么直接返回其属性值,如果找不到或报错了,无论如何没有达到预期的结果,那就调用 _getattr_ 方法。

_getattr_

  • 当以 点 属性名的形式访问属性时,如果属性不存在,则会执行对象的 _getattr_ 方法 该方法接受一个变量,item,即访问的属性名。返回值为本次获取的属性值,但是这个值并没有写入 对象的属性字典里。
  • 也就是说如果属性在__getattribute__中找到是不会执行这个方法的。
  • 这个方法也容易陷入无限的递归当中。

_setattr_

以点属性名的形式设置属性时,会调用 _setattr_ 方法,此方法需要将属性名和属性值的对应关系写入关系字典__dict__里。如果重写了该方法,一定不要忘记手动的更新 对象属性字典。

class Storage(object):

  def __init__(self, name):
    self.name = name  # 调用__setattr__方法

  def __getattribute__(self, item):  # 每个属性访问前都先调用该方法
    print('getattribute: %s' % item)
    ret = True
    if item == 'error':
      raise AttributeError(r'Error ~ "error"')  # 报错了依然执行~ 
    else:
      ret = object.__getattribute__(self, item)
    return ret

  def __getattr__(self, item):
    print('getattr: %s' % item)
    try:
      return self.__dict__[item]
    except (IndexError, KeyError)as e:
      print('No attribute %s ' % e)
    return '%s is error' % item

  def __setattr__(self, key, value):
    print('setattr: %s ' % key)
    self.__dict__.update({key:value})


file = Storage('file')
name = file.error  # 调用 __getattr__ 方法
# setattr: name 
# getattribute: __dict__
# getattribute: error
# getattr: error
# getattribute: __dict__
# No attribute 'error'

以上就是Python魔术方法专题的详细内容,更多关于Python 魔术方法的资料请关注三水点靠木其它相关文章!

Python 相关文章推荐
Python的Django框架中使用SQLAlchemy操作数据库的教程
Jun 02 Python
python决策树之CART分类回归树详解
Dec 20 Python
用tensorflow实现弹性网络回归算法
Jan 09 Python
python2.7无法使用pip的解决方法(安装easy_install)
Apr 03 Python
Python数据结构之图的应用示例
May 11 Python
Python变量访问权限控制详解
Jun 29 Python
解析python的局部变量和全局变量
Aug 15 Python
Python如何使用BeautifulSoup爬取网页信息
Nov 26 Python
pytorch构建多模型实例
Jan 15 Python
python matplotlib 绘图 和 dpi对应关系详解
Mar 14 Python
Python字符串对齐、删除字符串不需要的内容以及格式化打印字符
Jan 23 Python
python本地文件服务器实例教程
May 02 Python
关于Theano和Tensorflow多GPU使用问题
Jun 19 #Python
如何对python的字典进行排序
Jun 19 #Python
浅谈Python中的继承
Jun 19 #Python
python程序需要编译吗
Jun 19 #Python
python中round函数如何使用
Jun 19 #Python
keras实现theano和tensorflow训练的模型相互转换
Jun 19 #Python
Keras 切换后端方式(Theano和TensorFlow)
Jun 19 #Python
You might like
PHP使用curl函数发送Post请求的注意事项
2016/11/26 PHP
语义化 H1 标签
2008/01/14 Javascript
给Function做的OOP扩展
2009/05/07 Javascript
jQuery 使用手册(六)
2009/09/23 Javascript
jQuery EasyUI API 中文文档 DateTimeBox日期时间框
2011/10/16 Javascript
jQuery 翻牌或百叶窗效果(内容三秒自动切换)
2012/06/14 Javascript
jQuery计算textarea中文字数(剩余个数)的小程序
2013/11/28 Javascript
jQuery焦点图轮播特效代码分享(3款)
2015/09/05 Javascript
js实现数组和对象的深浅拷贝
2017/09/30 Javascript
JS实现利用两个队列表示一个栈的方法
2017/12/13 Javascript
VUE2.0+Element-UI+Echarts封装的组件实例
2018/03/02 Javascript
Vue异步组件处理路由组件加载状态的解决方案
2018/09/07 Javascript
对angular 实时更新模板视图的方法$apply详解
2018/10/09 Javascript
使用gulp构建前端自动化的方法示例
2018/12/25 Javascript
小程序云开发之用户注册登录
2019/05/18 Javascript
vue.js的状态管理vuex中store的使用详解
2019/11/08 Javascript
javascript实现多边形碰撞检测
2020/10/24 Javascript
[00:57]林俊杰助阵DOTA2亚洲邀请赛
2015/01/28 DOTA
Python利用Beautiful Soup模块创建对象详解
2017/03/27 Python
Python下载网络文本数据到本地内存的四种实现方法示例
2018/02/05 Python
使用python读取txt文件的内容,并删除重复的行数方法
2018/04/18 Python
利用python实现对web服务器的目录探测的方法
2019/02/26 Python
pyqt5使用按钮进行界面的跳转方法
2019/06/19 Python
python代码编写计算器小程序
2020/03/30 Python
Python with语句和过程抽取思想
2019/12/23 Python
打包PyQt5应用时的注意事项
2020/02/14 Python
python读取图像矩阵文件并转换为向量实例
2020/06/18 Python
学习Python爬虫的几点建议
2020/08/05 Python
python产生模拟数据faker库的使用详解
2020/11/04 Python
Django跨域请求原理及实现代码
2020/11/14 Python
Linux如何为某个操作添加别名
2015/02/05 面试题
政风行风建设责任书
2014/07/23 职场文书
护士医德考评自我评价
2015/03/03 职场文书
2015年幼儿园班务工作总结
2015/05/12 职场文书
民事诉讼答辩状范文
2015/05/21 职场文书
JavaScript 防篡改对象的用法示例
2021/04/24 Javascript