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动态监控日志内容的示例
Feb 16 Python
解决python3在anaconda下安装caffe失败的问题
Jun 15 Python
用python实现对比两张图片的不同
Feb 05 Python
Python实现查找最小的k个数示例【两种解法】
Jan 08 Python
Python实现账号密码输错三次即锁定功能简单示例
Mar 29 Python
Python增强赋值和共享引用注意事项小结
May 28 Python
python命令行参数用法实例分析
Jun 25 Python
基于python的itchat库实现微信聊天机器人(推荐)
Oct 29 Python
PyTorch 随机数生成占用 CPU 过高的解决方法
Jan 13 Python
基于Python3读写INI配置文件过程解析
Jul 23 Python
PyTorch中Tensor的数据类型和运算的使用
Sep 03 Python
python list的index()和find()的实现
Nov 16 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的十大要点(上)
2009/02/04 PHP
ajax+php打造进度条代码[readyState各状态说明]
2010/04/12 PHP
php中mysql操作buffer用法详解
2015/03/19 PHP
PHPMAILER实现PHP发邮件功能
2018/04/18 PHP
PHP+MySQL高并发加锁事务处理问题解决方法
2018/04/30 PHP
Laravel相关的一些故障解决
2020/08/19 PHP
jQuery读取和设定KindEditor值的方法
2013/11/22 Javascript
jQuery源码解读之removeAttr()方法分析
2015/02/20 Javascript
浅谈JavaScript中的Math.atan()方法的使用
2015/06/14 Javascript
初识angular框架后的所思所想
2016/02/19 Javascript
JS实现添加,替换,删除节点元素的方法
2016/06/30 Javascript
jQuery-mobile事件监听与用法详解
2016/11/23 Javascript
AngularJS基于provider实现全局变量的读取和赋值方法
2017/06/28 Javascript
Thinkjs3新手入门之添加一个新的页面
2017/12/06 Javascript
vue+element实现批量删除功能的示例
2018/02/28 Javascript
vue自动化表单实例分析
2018/05/06 Javascript
说说如何利用 Node.js 代理解决跨域问题
2019/04/22 Javascript
利用es6 new.target来对模拟抽象类的方法
2019/05/10 Javascript
微信小程序如何获取群聊的openGid以及名称详解
2019/07/17 Javascript
js实现随机圆与矩形功能
2020/10/29 Javascript
[01:55]TI9显影之尘系列 - Evil Geniuses
2019/08/22 DOTA
简单的连接MySQL与Python的Bottle框架的方法
2015/04/30 Python
python类和函数中使用静态变量的方法
2015/05/09 Python
python将字典列表导出为Excel文件的方法
2019/09/02 Python
python 爬取古诗文存入mysql数据库的方法
2020/01/08 Python
python 回溯法模板详解
2020/02/26 Python
python json 递归打印所有json子节点信息的例子
2020/02/27 Python
pycharm开发一个简单界面和通用mvc模板(操作方法图解)
2020/05/27 Python
Python绘图之二维图与三维图详解
2020/08/04 Python
Python3+RIDE+RobotFramework自动化测试框架搭建过程详解
2020/09/23 Python
html5指南-6.如何创建离线web应用程序实现离线访问
2013/01/07 HTML / CSS
旷课检讨书2000字
2014/01/14 职场文书
九月份红领巾广播稿
2014/01/22 职场文书
房地产公司工程部经理岗位职责
2015/04/09 职场文书
学困生帮扶工作总结
2015/08/13 职场文书
初三英语教学反思
2016/02/15 职场文书