Python的魔法方法 .
基本行为和属性
__init__(self[,....])构造函数 . 在实例化对象的时候会自动运行
__del__(self)析构函数 . 在对象被回收机制回收的时候会被调用
__str__(self)输出函数 . 在实例对象请求输出的时候会被调用.
__repr__(self). 当直接调用实例对象的时候会被调用
__new__(cls,[,...]). 她的第一个参数是这个类 , 其他的参数被直接传送到 __init__ . 并且__new__是一个对象实例化的时候所调用的第一个方法(所以可以在这里做点手脚) ,
__bool__(self) 定义当被bool类 调用的时候 应当返回的值
__len__(self) 定义单被len调用的时候的行为.
__hash__(self) 定义当被hash 调用的时候返回的函数 .
__getattr__ (self,name)定义当用户师徒访问一个不存在的属性时 所执行的行为 .
__getattribute__(self,name)定义当该属性被访问时的行为
__setattribute__ (self,name,value)定义一个属性被设置时的行为 .
__delattr__(self,name)定义一个属性被删除时的行为
__dir__(self) 定义dir被调用时的行为.
__get__(self,instance,owner) 定义当描述符 被取得时的行为
__set__(self,instance,owner)定义当描述符的值被改变是的行为.
__delete__(self,instance) 定义当描述符被删除时的行为
比较操作符,算术运算符,反运算,增量运算,一元操作,类型转换,上下文管理,容器类型.
详见http://bbs.fishc.com/thread-48793-1-2.html
当属性的名称和方法的名称一样的时候 , 属性的名称会自动覆盖方法的名称 , 所以属性和方法的名称影噶尽量分开 .
- import time as t
- import sys
- class MyTimer():
- def __init__(self): # 上去先设置 各种属性防止 使用不当出错 .
- self.unit=['年','月','日','小时','分钟','秒']
- self.prompt='未开始计时.'
- self.lasted=[]
- self.begin=0
- self.end=0
- def __str__(self):
- print('我被调用了 .')
- return self.prompt
- __repr__=__str__
- def start(self):
- self.begin=t.localtime()
- print('开始计时')
- def stop(self):
- if not self.begin:
- print("请先开始调用 start")
- else:
- self.end=t.localtime()
- self._calc()
- print('计时结束')
- def _calc(self):
- self.lasted=[]
- self.prompt='总共运行了'
- for index in range(6):
- self.lasted.append(self.end[index]-self.begin[index])
- if self.lasted[index]:
- self.prompt+=str(self.lasted[index])+self.unit[index]
- Python 3.5.2 (v3.5.2:4def2a2901a5, Jun 25 2016, 22:01:18) [MSC v.1900 32 bit (Intel)] on win32
- Type "copyright", "credits" or "license()" for more information.
- >>>
- =============== RESTART: C:/Users/Administrator/Desktop/new.py ===============
- >>> t1=MyTimer()
- >>> t1
- 我被调用了 .
- 未开始计时.
- >>> t1
- 我被调用了 .
- 未开始计时.
- >>> t1.Stop()
- 请先开始调用 start
- >>> t1.Start()
- 开始计时
- >>> t1.Stop()
- 计时结束
- >>> t1
- 我被调用了 .
- 总共运行了5秒
- >>> print(t1)
- 我被调用了 .
- 总共运行了5秒
- >>>
魔法方法的坑 , 和常见的躲坑方法 .
- # 这是一个 求面积的程序( 很显然有坑 ) , 如果属性名定义为 square的话 就默认 width=height
- class Rectangle:
- def __init__(self,width=0,height=0):
- self.width=width
- self.height=height
- def __setattr__(self,name,value):
- if name=='suqare':
- self.width=valuc
- self,height=value
- else:
- #super().__setattr__(name,value)
- self.name=value #如果程序是这样的话实例化对象的时候会出现无限递归 为啥呢?
- def GetArea(self): # 在 最初的 init 那里开始对一个不存在的属性进行赋值 然后就调用了 setattr 调用 setattr的时候 又有 对 该属性(不存在.)进行赋值 然后就递归了.
- return self.width*self.height # 解决的办法就是 在出错的地方使用官方提供的 方法 super().__setattr__(name,value)
- >>> c1=Rectangle()
- Traceback (most recent call last):
- File "<pyshell#0>", line 1, in <module>
- c1=Rectangle()
- File "C:\Users\Administrator\Desktop\new.py", line 3, in __init__
- self.width=width
- File "C:\Users\Administrator\Desktop\new.py", line 12, in __setattr__
- self.name=value #如果程序是这样的话实例化对象的时候会出现无限递归 为啥呢?
- File "C:\Users\Administrator\Desktop\new.py", line 12, in __setattr__
- self.name=value #如果程序是这样的话实例化对象的时候会出现无限递归 为啥呢?
- File "C:\Users\Administrator\Desktop\new.py", line 12, in __setattr__
- self.name=value #如果程序是这样的话实例化对象的时候会出现无限递归 为啥呢?
- File "C:\Users\Administrator\Desktop\new.py", line 12, in __setattr__
- self.name=value #如果程序是这样的话实例化对象的时候会出现无限递归 为啥呢?
- File "C:\Users\Administrator\Desktop\new.py", line 12, in __setattr__
- self.name=value #如果程序是这样的话实例化对象的时候会出现无限递归 为啥呢?
- ...
- ...
- ...
- File "C:\Users\Administrator\Desktop\new.py", line 12, in __setattr__
- self.name=value #如果程序是这样的话实例化对象的时候会出现无限递归 为啥呢?
- File "C:\Users\Administrator\Desktop\new.py", line 12, in __setattr__
- self.name=value #如果程序是这样的话实例化对象的时候会出现无限递归 为啥呢?
- RecursionError: maximum recursion depth exceeded while calling a Python object
- >>>
上面的坑 详见 源代码注释 .
躲开这些坑的办法就是 当你重写这些魔法方法之后 , 剩余的 就让原来的官方方法就解决 .
- # 这是一个 求面积的程序 , 如果属性名定义为 square的话 就默认 width=height
- class Rectangle:
- def __init__(self,width=0,height=0):
- self.width=width
- self.height=height
- def __setattr__(self,name,value):
- if name=='suqare':
- self.width=valuc
- self,height=value
- else:
- self.__dict__[name]=value
- #super().__setattr__(name,value)
- #self.name=value #如果程序是这样的话实例化对象的时候会出现无限递归 为啥呢?
- def GetArea(self): # 在 最初的 init 那里开始对一个不存在的属性进行赋值 然后就调用了 setattr 调用 setattr的时候 又有 对 该属性(不存在.)进行赋值 然后就递归了.
- return self.width*self.height # 解决的办法就是 在出错的地方使用官方提供的 方法 super().__setattr__(name,value)
- =============== RESTART: C:\Users\Administrator\Desktop\new.py ===============
- >>> r1=Rectangle(4,5)
- >>> r1.__dict__ # 将该实例化对象的 所有属性 打印出来
- {'height': 5, 'width': 4}
- >>> r1.GetArea()
- 20
- =============== RESTART: C:\Users\Administrator\Desktop\new.py ===============
- >>> r1=Rectangle(4,5)
- >>> r1.__dict__
- {'width': 4, 'height': 5}
- >>> # 显然是可以的 .
Python的魔法方法 .的更多相关文章
- python之魔法方法介绍
1.1. 简介 什么是魔法方法呢?它们在面向对象的Python的处处皆是.它们是一些可以让你对类添加“魔法”的特殊方法. 它们经常是两个下划线包围来命名的(比如 __init__ , __lt__ ) ...
- 【Python】 魔法方法
魔法方法 这个名字真的很中二有没有 = =(或者说翻译气息太浓了,作为一个学外语的看到这种真是想吐槽的不行..) 从形式上来说,在方法的名字前后个加上两条下划线的就是魔法方法了 .从功能上说,所有魔法 ...
- python,魔法方法指南
1.简介 本指南归纳于我的几个月的博客,主题是 魔法方法 . 什么是魔法方法呢?它们在面向对象的Python的处处皆是.它们是一些可以让你对类添加“魔法”的特殊方法. 它们经常是两个下划线包围来命名的 ...
- python中魔法方法__init__,__str__,__del__的详细使用方法
1. python中的魔法方法, 类似__init__, __str__等等,这些内置好的特定的方法进行特定的操作时会自动被调用 2. __init__的使用方法 class 类名(object): ...
- 21 python的魔法方法(转)
魔法方法 含义 基本的魔法方法 __new__(cls[, ...]) 1. __new__ 是在一个对象实例化的时候所调用的第一个方法2. 它的第一个参数是这个类,其他的参数是用来直接传递给 _ ...
- Python的魔法方法??
就是可以给你的类增加魔力的特殊方法,如果你的对象实现 (重载)了这些方法中的某一个,那么这个方法就会在特殊的情况下被 Python 所调用,你可以定义自己想要的行为,而这一切都是自动发生的. __in ...
- python基础--魔法方法、迭代器、上下文管理
isinstance:判断一个对象是否是某个类的实例 参数一:要判断的对象 参数二:要判断的类型 issubclass:判断一个类是否是另一个类的子类 参数一:是待判断的子类 参数二:待判断的父类 _ ...
- Python随笔--魔法方法(析构与构造)
#析构方法的调用
- python中魔法方法(持续更新)
1.对于一个自定义的类,如果实现了 __call__ 方法,那么该类的实例对象的行为就是一个函数,是一个可以被调用(callable)的对象.例如: class Add: def __init__(s ...
随机推荐
- ArcGIS 10.3 安装及破解
系统环境:win7 64位操作系统. 一.ArcGIS 10.3包简介 ArcGIS 10.3 下载包含 1. ArcGIS for Desktop ArcGIS for Desktop简介: Ar ...
- VMware Workstation 12 Pro虚拟机下载(含序列号)
VMware Workstation 12 Pro 官网下载地址:http://www.vmware.com/products/workstation/workstation-evaluation V ...
- 正确理解Spring AOP中的Around advice
Spring AOP中,有Before advice和After advice,这两个advice从字面上就可以很容易理解,但是Around advice就有点麻烦了. 乍一看好像是Before ad ...
- Unity Invoke 方法
Invoke() 方法是 Unity3D 的一种委托机制 如: Invoke("a", 5); 它的意思是:5 秒之后调用 a() 方法: 使用 Invoke() 方法需要注意 ...
- Uva---10881 Piotr's Ants(蚂蚁)
Problem DPiotr's AntsTime Limit: 2 seconds "One thing is for certain: there is no stopping them ...
- URL和搜索引擎优化
URL要短:谷歌的算法是给5个以后的单词更少的权重 URL不含和少含问号:防止死循环 小写字母:搜索引擎遵守HTTP规范,区分大小写.
- jquery mobile 请求数据方法执行时显示加载中提示框
在jquery mobile开发中,经常需要调用ajax方法,异步获取数据,如果异步获取数据方法由于网速等等的原因,会有一个反应时间,如果能在点击按钮后数据处理期间,给一个正在加载的提示,客户体验会更 ...
- [原创]VM虚拟机安装centos6.4详细图文教程
1.启动虚拟机,新建虚拟机. 2.选择从镜像安装,选择centos6.4的路径. 3.设置用户名和密码.注:只能用小写字母. 4.选择安装路径. 5.配置磁盘大小. 6.准备创建. 如果需要自定义 ...
- XPATH 带命名空间数据的读取
在XML中,很多情况下有命名空间,如果直接使用XPATH 读取是会读到空节点. 解决办法如下: InputStream is=loader.getResourceAsStream("com/ ...
- Ubuntu里面的安装命令总结
本人是新手中的新手,才开始用ubuntu.下面,总结一下安装软件的方法...... 0. 利用apt-get 其实,在ubuntu下安装软件的方法其实灰常简单.就是在终端里面输入: sudo apt- ...