Python中的数学运算操作符使用进阶


Posted in Python onJune 20, 2016

Python中对象的行为是由它的类型 (Type) 决定的。所谓类型就是支持某些特定的操作。数字对象在任何编程语言中都是基础元素,支持加、减、乘、除等数学操作。
Python的数字对象有整数和浮点数,支持各种数学操作,比如+, -,*, /等。 没有这些操作符,程序中只能使用函数调用的方式进行数学运算,比如add(2, 3), sub(5, 2)。
程序中操作符的作用与普通数学操作的用法是一致的,使用中更加简便直观。Python中,这些操作符实现是通过定义一些object的特殊方法实现的,比如object.__add__()和object.__sub__()。如果用户在自己定义类时实现上述特殊方法,可以使自定义类的对象支持相应的数学操作,从而模拟数字对象的行为。这其实是达到了操作符重载的效果。

这里通过实现一个具有支持加法运算的中文数字类说明如何在Python中实现一个支持常见的数学操作。ChineseNumber类的基本定义如下。

class ChineseNumber:
  def __init__(self, n):
    self.num = n
    self.alphabet = [u'零', u'一', u'二', u'三', u'四', 
      u'五', u'六', u'七', u'八', u'九', u'十']

  def __str__(self):
    sign = '负' if self.num < 0 else ''
    return sign + ''.join([self.alphabet[int(s)] for s in str(abs(self.num))])

  def __repr__(self):
    return self.__str__()

目前,实现的效果是这样的:

>>> a = ChineseNumber(2)
>>> a  #调用a.__repr__()
二
>>> print(a)  #调用a.__str__()

二

一般数学操作符
定义类时,实现__add__()方法,可以给这个类增加+操作符。给ChineseNumber增加如下方法:

def __add__(self, other):
    if type(other) is ChineseNumber:
      return ChineseNumber(self.num + other.num)
    elif type(other) is int:
      return ChineseNumber(self.num + other)
    else:
      return NotImplemented

这时ChineseNumber的对象就可以使用+了。

>>> a, b = ChineseNumber(2), ChineseNumber(10)
>>> a + b
十二
>>> a + 5
七
>>> a + 3.7
TypeError: unsupported operand type(s) for +: 'ChineseNumber' and 'float'

对于+,a + b相当于调用a.__add__(b). 类似地,可以定义其他数学操作符,见下表。

object.__add__(self, other): +
object.__sub__(self, other): -
object.__mul__(self, other): *
object.__matmul__(self, other): @
object.__truediv__(self, other): /
object.__floordiv__(self, other): //
object.__mod__(self, other): %
object.__divmod__(self, other): divmod, divmod(a, b) = (a/b, a%b)
object.__pow__(self, other[,modulo]): **, pow()
object.__lshift__(self, other): <<
object.__rshift__(self, other): >>
object.__and__(self, other): &
object.__xor__(self, other): ^
object.__or__(self, other): |

操作数反转的数学操作符 (Reflected/Swapped Operand)

>>> 2 + a
TypeError: unsupported operand type(s) for +: 'int' and 'ChineseNumber'

2是整数类型,它的__add__()方法不支持ChineseNumber类的对象,所以出现了上述错误。定义操作数反转的数学操作符可以解决这个问题。给ChineseNumber类添加__radd__()方法,实现操作数反转的+运算。

def __radd__(self, other):
    return self.__add__(other)

对于a + b,如果a没有定义__add__()方法,Python尝试调用b的__radd__()方法。此时,a + b相当于调用b.__radd__(a)。

>>> a = ChineseNumber(2)
>>> 2 + a
四

类似地,可以定义其他操作数反转的数学操作符,见下表。

object.__radd__(self, other): +
object.__rsub__(self, other): -
object.__rmul__(self, other): *
object.__rmatmul__(self, other): @
object.__rtruediv__(self, other): /
object.__rfloordiv__(self, other): //
object.__rmod__(self, other): %
object.__rdivmod__(self, other): divmod, divmod(a, b) = (b/a, b%a)
object.__rpow__(self, other[,modulo]): **, pow()
object.__rlshift__(self, other): <<
object.__rrshift__(self, other): >>
object.__rand__(self, other): &
object.__rxor__(self, other): ^
object.__ror__(self, other): |

运算赋值操作符
运算赋值操作符使用单个操作符完成运算和赋值操作,比如a += b相当于调用a = a + b。为ChineseNumber增加__iadd__()方法,可以实现+=操作符。

def __iadd__(self, other):
    if type(other) is ChineseNumber:
      self.num += other.num
      return self
    elif type(other) is int:
      self.num += other
      return self
    else:
      return NotImplemented

此时,

>>> a, b = ChineseNumber(2), ChineseNumber(10)
>>> a += b
>>> a
十二
>>> a + 7
>>> a
十九

类似地,可以定义其他运算赋值操作符,见下表。

object.__iadd__(self, other): +=
object.__isub__(self, other): -=
object.__imul__(self, other): *=
object.__imatmul__(self, other): @=
object.__itruediv__(self, other): /=
object.__ifloordiv__(self, other): //=
object.__imod__(self, other): %=
object.__ipow__(self, other[,modulo]): **=
object.__ilshift__(self, other): <<=
object.__irshift__(self, other): >>=
object.__iand__(self, other): &=
object.__ixor__(self, other): ^=
object.__ior__(self, other): |=

一元数学操作符
一元数学操作符是只有一个操作数的运算,比如取负数的操作符-。-对应的特殊函数是__neg__()。为ChineseNumber添加__neg__()方法,

def __neg__(self):
    return ChineseNumber(-self.num)

此时,ChineseNumber对象就支持-操作了。

>>> a = ChineseNumber(5)
>>> -a
负五

其他一元运算符见下表。

object.__neg__(self): -
object.__pos__(self): +
object.__abs__(self): abs()
object.__invert__(self): ~
object.__complex__(self): complex()
object.__int__(self): int()
object.__float__(self): float()
object.__round__(self): round()
object.__index__(self): operator.index()
Python 相关文章推荐
Python入门篇之条件、循环
Oct 17 Python
详解Python中的type()方法的使用
May 21 Python
python开发之list操作实例分析
Feb 22 Python
Python实现输出程序执行进度百分比的方法
Sep 16 Python
django项目运行因中文而乱码报错的几种情况解决
Nov 07 Python
python+pandas生成指定日期和重采样的方法
Apr 11 Python
详解Python函数式编程—高阶函数
Mar 29 Python
解决pyCharm中 module 调用失败的问题
Feb 12 Python
如何使用Python抓取网页tag操作
Feb 14 Python
Django 允许局域网中的机器访问你的主机操作
May 13 Python
python中温度单位转换的实例方法
Dec 27 Python
用Python实现屏幕截图详解
Jan 22 Python
Python中在for循环中嵌套使用if和else语句的技巧
Jun 20 #Python
解析Python中的生成器及其与迭代器的差异
Jun 20 #Python
Python判断列表是否已排序的各种方法及其性能分析
Jun 20 #Python
Python编程中装饰器的使用示例解析
Jun 20 #Python
12步入门Python中的decorator装饰器使用方法
Jun 20 #Python
深入学习Python中的装饰器使用
Jun 20 #Python
Python中Iterator迭代器的使用杂谈
Jun 20 #Python
You might like
PHP获取服务器端信息的方法
2014/11/28 PHP
Zend Framework教程之模型Model基本规则和使用方法
2016/03/04 PHP
PHP实现负载均衡的加权轮询方法分析
2018/08/22 PHP
PHP的PDO事务与自动提交
2019/01/24 PHP
Javascript表达式中连续的 &amp;&amp; 和 || 之赋值区别
2010/10/17 Javascript
jquery EasyUI的formatter格式化函数代码
2011/01/12 Javascript
javascipt匹配单行和多行注释的正则表达式
2013/11/20 Javascript
JavaScript实现Flash炫光波动特效
2015/05/14 Javascript
js实现获取当前时间是本月第几周的方法
2015/08/11 Javascript
javascript实现自动输出文本(打字特效)
2015/08/27 Javascript
JS与Ajax Get和Post在使用上的区别实例详解
2016/06/08 Javascript
Jquery实时监听input value的实例
2017/01/26 Javascript
Vue.js组件间的循环引用方法示例
2017/12/27 Javascript
vue实现word,pdf文件的导出功能
2018/07/31 Javascript
实例详解ztree在vue项目中使用并且带有搜索功能
2018/08/24 Javascript
element ui table 增加筛选的方法示例
2018/11/02 Javascript
浅谈Express.js解析Post数据类型的正确姿势
2019/05/30 Javascript
javaScript中indexOf用法技巧
2019/11/26 Javascript
node事件循环和process模块实例分析
2020/02/14 Javascript
微信小程序学习总结(五)常见问题实例小结
2020/06/04 Javascript
Python实现的数据结构与算法之快速排序详解
2015/04/22 Python
python在线编译器的简单原理及简单实现代码
2018/02/02 Python
Python Web程序部署到Ubuntu服务器上的方法
2018/02/22 Python
python3 图片referer防盗链的实现方法
2018/03/12 Python
tensorflow 1.0用CNN进行图像分类
2018/04/15 Python
Python 利用scrapy爬虫通过短短50行代码下载整站短视频
2018/10/29 Python
python自动化之Ansible的安装教程
2019/06/13 Python
Python3+PyInstall+Sciter解决报错缺少dll、html等文件问题
2019/07/15 Python
18个Python脚本可加速你的编码速度(提示和技巧)
2019/10/17 Python
Python3实现个位数字和十位数字对调, 其乘积不变
2020/05/03 Python
使用K.function()调试keras操作
2020/06/17 Python
Stuart Weitzman欧盟:美国奢华鞋履品牌
2017/05/24 全球购物
医学检验专业个人求职信范文
2013/12/04 职场文书
教师作风整顿个人剖析材料
2014/10/10 职场文书
单位单身证明样本
2014/10/11 职场文书
2015年电话客服工作总结
2015/05/18 职场文书