Python进阶学习之特殊方法实例详析


Posted in Python onDecember 01, 2017

前言

最近在学习python,学习到了一个之前没接触过的--特殊方法。

什么是特殊方法?当我们在设计一个类的时候,python中有一个用于初始化的方法$__init__$,类似于java中的构造器,这个就是特殊方法,也叫作魔术方法。简单来说,特殊方法可以给你设计的类加上一些神奇的特性,比如可以进行python原生的切片操作,迭代、连乘操作等。在python中,特殊方法以双下划线开始,以双下划线结束。

一个大例子

数学中有一个表示数的概念叫做向量,但是python中的数据类型却没有。我们来设法用python实现它。

首先考虑,向量跟普通的数据类型不同,传统的数可以直接进行运算,向量则需要对不同的坐标分别运算。来试试。

首先定义一个类,实现初始化方法。

# 实现向量类型
class Vector:
 
 def __init__(self, x=0, y=0):
  self.x = x
  self.y = y

如何实现向量的加法?二维向量中,向量的加法就是每个坐标分别相加得到的结果。在python中有个$__add__$方法,用来进行加法操作。

class Vector:
 
 def __init__(self, x=0, y=0):
  self.x = x
  self.y = y

 # 实现向量加法
 def __add__(self, other):
  x = self.x + other.x
  y = self.y + other.y
  return Vector(x, y)

我们对x和y变量分别进行相加,然后返回Vector。在python你可以对字符串直接用加法拼接起来的原理就在此,python实现了针对字符串的add方法。

实现了加法,乘法的道理一样,分别对每个坐标单独相乘即可。

class Vector:
 
 def __init__(self, x=0, y=0):
  self.x = x
  self.y = y

 # 实现向量加法
 def __add__(self, other):
  x = self.x + other.x
  y = self.y + other.y
  return Vector(x, y)
 
 # 实现向量乘法,例如r*3
 def __mul__(self, scalar):
  return Vector(self.x*scalar, self.y*scalar)

我们在进行向量运算时还有一个常用的操作是求向量的模,我们用$__abs__$特殊方法来实现,abs一般用来求一个数的绝对值,向量用不到,用来求模刚好合适。使用math模块中的hypot方法计算$\sqrt(x^2+y^2)$。

class Vector:
 
 def __init__(self, x=0, y=0):
  self.x = x
  self.y = y
 
 # 真假值,如果向量模为0,返回false
 def __bool__(self):
  return bool(abs(self))

 # 实现向量加法
 def __add__(self, other):
  x = self.x + other.x
  y = self.y + other.y
  return Vector(x, y)
 
 # 实现向量乘法,例如r*3
 def __mul__(self, scalar):
  return Vector(self.x*scalar, self.y*scalar)

 
 # 返回向量的模
 # hypot()返回欧几里德范数 sqrt(x*x + y*y)
 def __abs__(self):
  return hypot(self.x, self.y)

找个例子运行下。

v = Vector(2, 3)
print(v)
v2 = Vector(4, 5)
print(v+v2)
print(v+v2*2)
<__main__.Vector object at 0x000002B4B1843C50>
<__main__.Vector object at 0x000002B4B1843EF0>
<__main__.Vector object at 0x000002B4B1843898>

可以运行了,貌似是正确的,但是输出的结果很奇怪。怎么办?python中有个$__repr__$特殊方法,可以修改控制台输出的样式。

class Vector:
 
 def __init__(self, x=0, y=0):
  self.x = x
  self.y = y
 
 # 真假值,如果向量模为0,返回false
 def __bool__(self):
  return bool(abs(self))

 # 实现向量加法
 def __add__(self, other):
  x = self.x + other.x
  y = self.y + other.y
  return Vector(x, y)
 
 # 实现向量乘法,例如r*3
 def __mul__(self, scalar):
  return Vector(self.x*scalar, self.y*scalar)
 
 # 返回向量的模
 # hypot()返回欧几里德范数 sqrt(x*x + y*y)
 def __abs__(self):
  return hypot(self.x, self.y)
 
 # 实现__repr__方法,在控制台打印向量时会输出Vector(1, 2)
 # 实现__str__,使用str()返回字符串
 def __repr__(self):
  return 'Vector(%r, %r)' % (self.x, self.y)

实现了$__repr__$方法,我们就可以在控制台输出Vecotor(x,y)。与之对应的有个$__str__$方法,使用str()返回相应的字符串,展示给用户。

现在来看下之前程序运行的结果。

v = Vector(2, 3)
print(v)
v2 = Vector(4, 5)
print(v+v2)
print(v+v2*2)
print(abs(v))
Vector(2, 3)
Vector(6, 8)
Vector(10, 13)
3.605551275463989

效果不错。

通过实现特殊方法,自定义类型可以表现的跟内置类型一样,让我们能够写出更具有python风格的代码。

除了上面说到的几个特殊方法外,python还有差不多80多个特殊方法,比如$__len__$方法可以用来求长度,$__getitem__$可以使用haha[2]之类的操作进行切片和迭代等,同样的还有$__setitem__$。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对三水点靠木的支持。

Python 相关文章推荐
python实现的防DDoS脚本
Feb 08 Python
Python时间获取及转换知识汇总
Jan 11 Python
关于python2 csv写入空白行的问题
Jun 22 Python
Django2.1集成xadmin管理后台所遇到的错误集锦(填坑)
Dec 20 Python
Python简单处理坐标排序问题示例
Jul 11 Python
用Python实现二叉树、二叉树非递归遍历及绘制的例子
Aug 09 Python
用Pytorch训练CNN(数据集MNIST,使用GPU的方法)
Aug 19 Python
详解django中Template语言
Feb 22 Python
PyQt5中QTableWidget如何弹出菜单的示例代码
Feb 23 Python
python实现简单井字棋游戏
Mar 04 Python
Python 程序员必须掌握的日志记录
Aug 17 Python
Python中requests做接口测试的方法
May 30 Python
Python用户推荐系统曼哈顿算法实现完整代码
Dec 01 #Python
浅谈python 里面的单下划线与双下划线的区别
Dec 01 #Python
vscode 远程调试python的方法
Dec 01 #Python
Python中单、双下划线的区别总结
Dec 01 #Python
从CentOS安装完成到生成词云python的实例
Dec 01 #Python
Django的分页器实例(paginator)
Dec 01 #Python
浅谈python装饰器探究与参数的领取
Dec 01 #Python
You might like
Thinkphp5.0框架视图view的循环标签用法示例
2019/10/12 PHP
JS正则中的RegExp对象对象
2012/11/07 Javascript
ajax异步刷新实现更新数据库
2012/12/03 Javascript
JS鼠标滑过图片时切换图片实现思路
2013/09/12 Javascript
js特殊字符转义介绍
2013/11/05 Javascript
jquery插件tooltipv顶部淡入淡出效果使用示例
2013/12/05 Javascript
JQuery中$.ajax()方法参数详解及应用
2013/12/12 Javascript
javascript教程之不完整的继承(js原型链)
2014/01/13 Javascript
Node.js和MongoDB实现简单日志分析系统
2015/04/25 Javascript
jQuery实现为图片添加镜头放大效果的方法
2015/06/25 Javascript
jquery中validate与form插件提交的方式小结
2016/03/26 Javascript
javascript中利用柯里化函数实现bind方法
2016/04/29 Javascript
js判断复选框是否选中及选中个数的实现代码
2016/05/30 Javascript
Ionic 2 实现列表滑动删除按钮的方法
2017/01/22 Javascript
Easyui在treegrid添加控件的实现方法
2017/06/23 Javascript
Vue.js基础指令实例讲解(各种数据绑定、表单渲染大总结)
2017/07/03 Javascript
element-ui 关于获取select 的label值方法
2018/08/24 Javascript
vue子路由跳转实现tab选项卡
2019/07/24 Javascript
微信小程序 可搜索的地址选择实现详解
2019/08/28 Javascript
JavaScript setTimeout()基本用法有哪些
2020/11/04 Javascript
在Python下尝试多线程编程
2015/04/28 Python
使用Python简单的实现树莓派的WEB控制
2016/02/18 Python
如何用Python来搭建一个简单的推荐系统
2019/08/07 Python
Python使用指定字符长度切分数据示例
2019/12/05 Python
Python将list元素转存为CSV文件的实现
2020/11/16 Python
python list的index()和find()的实现
2020/11/16 Python
美国一家专业的太阳镜网上零售商:Solstice太阳镜
2016/07/25 全球购物
中学生学习生活的自我评价
2013/10/26 职场文书
企业授权委托书范本
2014/04/02 职场文书
产品陈列协议书(标准版)
2014/09/17 职场文书
护士年终考核评语
2014/12/31 职场文书
毕业生评语大全
2015/01/04 职场文书
万能检讨书
2015/01/27 职场文书
务工证明怎么写
2015/06/18 职场文书
Vue3.0 手写放大镜效果
2021/07/25 Vue.js
Oracle配置dblink访问PostgreSQL的操作方法
2022/03/21 PostgreSQL