python魔法方法-反射运算和增量运算
反射运算
什么是反射运算符,其实就是反转了两个对象,下面先看一个普通运行符的实现:
class Foo(object):
def __init__(self, x):
self.x = x def __add__(self, other):
return 'Foo:%s + %s' % (self.x, other.x) class Boo(object):
def __init__(self, x):
self.x = x def __add__(self, other):
return 'Boo:%s + %s' % (self.x, other.x) a = Foo(123)
b = Boo(321)
print a + b
print b + a
在普通的加法运算中,调用的是+号左边的__add__方法,调用谁谁就为self。所以左边是self,右边为other,所以结果如上。
而反射运行其实就是交换这两者,下面看例子:
class Foo(object):
def __init__(self, x):
self.x = x def __radd__(self, other):
return 'Foo:%s + %s' % (self.x, other.x) class Boo(object):
def __init__(self, x):
self.x = x def __radd__(self, other):
return 'Boo:%s + %s' % (self.x, other.x) a = Foo(123)
b = Boo(321)
print a + b
print b + a
首先,不同的地方是这里调用的+后右边的__radd__方法。然后本来是左边的为self的,现在变成了右边的为self。
总结起来就是:普通的运算调用的是运算符左边的方法,而反射运算符调用的是右边的方法,调用的是谁的方法,谁就为self。
这里有几点要注意的地方:
1.不支持同一个类的实例进行反射运算:
class Foo(object):
def __init__(self, x):
self.x = x def __radd__(self, other):
return 'Foo:%s + %s' % (self.x, other.x) a = Foo(123)
b = Foo(321)
print a + b
print b + a
2.当一个类实现了__add__的时候,将会掩盖__radd__方法,也就是__add__的优先度更高:
class Foo(object):
def __init__(self, x):
self.x = x def __radd__(self, other):
return 'Foo:%s + %s' % (self.x, other.x) class Boo(object):
def __init__(self, x):
self.x = x def __add__(self, other):
return 'Boo add:%s + %s' % (self.x, other.x) def __radd__(self, other):
return 'Boo radd:%s + %s' % (self.x, other.x) a = Foo(123)
b = Boo(321)
print a + b
print b + a
其逻辑如下:
首先a + b,python看到了 a 中没有 __add__方法(忽略了__radd__),就去 b 中找__radd__(而不是__add__),因为在右边找的时候,就意味要使用反射运算了。所以最后得到了这个结果。
然后是b + a,python看到了 b 中有 __add__方法,就直接调用了它,不管 a 的内部是如何的。
基本反射运算就是这么一回事,下面是一些总结:
__radd__(self, other)
反射加法
__rsub__(self, other)
反射减法的
__rmul__(self, other)
反射除法
__rfloordiv__(self, other)
反射地板除,使用//运算符的
__rdiv__(self, other)
反射除法,使用/运算符的.
__rtruediv__(self, other)
反射真除.注意只有from __future__ import division 的时候它才有效
__rmod__(self, other)
反射取模运算,使用%运算符.
__rdivmod__(self, other)
长除法,使用divmod()内置函数,当divmod(other,self)时被调用.
__rpow__
反射乘方,使用**运算符的
__rlshift__(self, other)
反射左移,使用<<操作符.
__rrshift__(self, other)
反射右移,使用>>操作符.
__rand__(self, other)
反射位与,使用&操作符.
__ror__(self, other)
反射位或,使用|操作符.
__rxor__(self, other)
反射异或,使用^操作符.
增量运算
所谓的增量运算,其实就是 x += 1 这样的形式,下面是几个例子:
class Foo(object):
def __init__(self, x):
self.x = x def __iadd__(self, other):
return 'Foo iadd: %s + %s' % (self.x, other) a = Foo(123)
a += 1
print a
但是,如果两个对象的实现了__iadd__,情况就会大为不同:
class Foo(object):
def __init__(self, x):
self.x = x def __iadd__(self, other):
return 'Foo iadd: %s + %s' % (self.x, other.x) class Boo(object):
def __init__(self, x):
self.x = x def __iadd__(self, other):
return 'Boo iadd: %s + %s' % (self.x, other.x) a = Foo(123)
b = Boo(321)
a += b
print a
看似很正常,然而代码如下时:
a = Foo(123)
b = Boo(321)
a += b
print a
b += a
print b
报错显示:str没有x这个属性,但是按照代码来看,两个对象都有x属性呀。
在b += a 这行有错,也就是说self为 b,other为 a。后来试验了一番,发现将:
return 'Boo iadd: %s + %s' % (self.x, other.x)
改为:
return 'Boo iadd: %s + %s' % (self.x, other)
代码就不会报错了,但是输出几个如下:
很奇怪,other变成了a中__iadd__的返回值了,也就是说当a调用了__iadd__方法之后,在将其用在其他的增量运算时,other不在代表a对象本身,而是其__iadd__的返回值。
当我们回归其本质:x += 1 ==> x = x + 1 可以看出,x 其实进行了重新赋值,重新赋值成了 __iadd__ 的返回值。而我们代码示例中,这个方法的返回值是一个字符串。在一开始时,x是我们类的实例。但是在进行了增量运算后,x 变成了魔法方法的返回值,也就是字符串了,所以才会出现以上的报错。
所以我们在使用的时候要注意 x 身份的改变,不然会有许多意想不到的麻烦。
关于增量方法的总结:
__iadd__(self, other)
加法赋值
__isub__(self, other)
减法赋值.
__imul__(self, other)
乘法赋值
__ifloordiv__(self, other)
整除赋值,地板除,相当于 //= 运算符.
__idiv__(self, other)
除法赋值,相当于 /= 运算符.
__itruediv__(self, other)
真除赋值,注意只有你 whenfrom __future__ import divisionis,才有效.
__imod_(self, other)
模赋值,相当于 %= 运算符.
__ipow__
乘方赋值,相当于 **= 运算符.
__ilshift__(self, other)
左移赋值,相当于 <<= 运算符.
__irshift__(self, other)
左移赋值,相当于 >>= 运算符.
__iand__(self, other)
与赋值,相当于 &= 运算符.
__ior__(self, other)
或赋值,相当于 |= 运算符.
__ixor__(self, other)
异或运算符,相当于 ^= 运算符.
欢迎大家交流。
参考资料:戳这里
python魔法方法-反射运算和增量运算的更多相关文章
- python魔法方法大全
1.python魔法方法详解: python魔法方法是可以修改重载的,如果你的对象实现(重载)了这些方法中的某一个,那么这个方法就会在特殊的情况下被 Python 所调用,你可以定义自己想要的行为,而 ...
- with上下文管理 python魔法方法
with语法在Python里很常见, 主要的利好是使用代码更简洁. 常见的使用场景有: 1. 资源对象的获取与释放. 使用with可以简化try...finally ... 2. 在不修改函数代码的前 ...
- python 魔法方法诠释
什么是Python魔法方法 什么是魔法方法呢?它们在面向对象的Python的处处皆是.它们是一些可以让你对类添加"魔法"的特殊方法. 它们经常是两个下划线包围来命名的(比如 ini ...
- Python魔法方法总结及注意事项
1.何为魔法方法: Python中,一定要区分开函数和方法的含义: 1.函数:类外部定义的,跟类没有直接关系的:形式: def func(*argv): 2.方法:class内部定义的函数(对象的方法 ...
- python 魔法方法补充(__setattr__,__getattr__,__getattribute__)
python 魔法方法补充 1 getattribute (print(ob.name) -- obj.func())当访问对象的属性或者是方法的时候触发 class F(object): def _ ...
- 1. Python 魔法方法
Python 魔法方法 基础: 如果你想... 所以,你写... Python调用... 初始化一个实例 x = MyClass() x.__init__() 作为一个字符串的"官方&quo ...
- python魔法方法:__getattr__,__setattr__,__getattribute__
python魔法方法:__getattr__,__setattr__,__getattribute__ 难得有时间看看书....静下心来好好的看了看Python..其实他真的没有自己最开始想的那么简单 ...
- python魔法方法-单目运算及一般算数运算
在比较的魔法方法中,我们讨论了魔法方法其实就是重载了操作符,例如>.<.==等.而这里,我们继续讨论有关于数值的魔法方法. 1.单目运算符或单目运算函数 __pos__(self) 实现一 ...
- python 魔法方法
I am not a creator, I just a porter. Note: Everything is object in python. 对于Python来说一切都是对象,也就是函数的参数 ...
随机推荐
- dubbo系列三、架构介绍及各模块关系
一.整体设计 图例说明: 图中左边淡蓝背景的为服务消费方使用的接口,右边淡绿色背景的为服务提供方使用的接口,位于中轴线上的为双方都用到的接口. 图中从下至上分为十层,各层均为单向依赖,右边的黑色箭头代 ...
- Windows10 中在指定目录下启动Powershell
(1)首先进入该目录: (2)按住shift键,且同时在该目录空白处鼠标右击,打开右键菜单: (3)此时可以发现,在右键菜单中,多了一项,叫做[在此处打开Powershell窗口(s)],点击该项: ...
- MySQL数据库——安装教程(5.7版本)
一.配置MySQL数据库 1.解压绿色版mysql,并改名为mysql5.7,如下图 对比一下下图5.6以前的版本,少data目录(存放数据)和my-default.ini文件(配置信息) 二.安装服 ...
- mysql通过centos本地命令行还原数据库出现乱码问题
将sql文件上传到centos系统中,还原mysql数据库,发现是乱码 mysql -h10.11.8.62 -uroot -p dbtest </data/dbsql/dbtest.sql 数 ...
- 解决Linux安装 VMware tools 工具的方法
一:启动linux服务器,并用远程登录工具访问linux服务器 1:启动系统 2:用服务器控制台 :查看点ip地址 3:用客户端 连接服务器 二:挂起 vm虚拟机的 tools 安装光盘 三:开始 ...
- confusion_matrix(混淆矩阵)
作者:十岁的小男孩 凡心所向,素履可往 目录 监督学习—混淆矩阵 是什么?有什么用?怎么用? 非监督学习—匹配矩阵 混淆矩阵 矩阵每一列代表预测值,每一行代表的是实际的类别.这个名字来源于它可以非常容 ...
- hdu3486 ST表区间最值+二分
还是挺简单的,但是区间处理的时候要注意一下 #include<iostream> #include<cstring> #include<cstdio> #inclu ...
- python 全栈开发,Day107(CRM初始,权限组件之权限控制,权限系统表设计)
一.CRM初始 CRM,客户关系管理系统(Customer Relationship Management).企业用CRM技术来管理与客户之间的关系,以求提升企业成功的管理方式,其目的是协助企业管理销 ...
- 【APUE | 7】命令形参、函数
命令形参 命令行参数是使用main()函数参数来处理的,其中,argc是指传入参数的个数,argv[]是一个指针数组,指向传递给程序的每个参数. 应当指出的是, argv[0]存储程序的名称,argv ...
- [转] web无插件播放RTSP摄像机方案,拒绝插件,拥抱H5!
需求 问题:有没有flash播放RTSP的播放器?H5能不能支持RTSP播放? 答案:没见过,以后估计也不会有: 问题:可以自己做浏览器插件播放RTSP吗? 答案:可以的,chrome做ppapi插件 ...