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中的二叉树查找算法模块使用指南
Jul 04 Python
python获取外网IP并发邮件的实现方法
Oct 01 Python
Python实现在某个数组中查找一个值的算法示例
Jun 27 Python
PyCharm设置护眼背景色的方法
Oct 29 Python
Python函数参数匹配模型通用规则keyword-only参数详解
Jun 10 Python
pyqt 实现为长内容添加滑轮 scrollArea
Jun 19 Python
python3 循环读取excel文件并写入json操作
Jul 14 Python
Python 利用OpenCV给照片换底色的示例代码
Aug 03 Python
python用700行代码实现http客户端
Jan 14 Python
Python爬虫入门案例之爬取二手房源数据
Oct 16 Python
Python OpenGL基本配置方式
May 20 Python
Golang Web 框架Iris安装部署
Aug 14 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
php输入数据统一类实例
2015/02/23 PHP
laravel 5.4中实现无限级分类的方法示例
2017/07/27 PHP
在laravel中使用with实现动态添加where条件
2019/10/10 PHP
IE和Firefox下event事件杂谈
2009/12/18 Javascript
dreamweaver 安装Jquery智能提示
2011/04/02 Javascript
window.event快达到全浏览器支持了,以后使用就方便了
2011/11/30 Javascript
EasyUI的treegrid组件动态加载数据问题的解决办法
2011/12/11 Javascript
Jquery动态替换div内容及动态展示的方法
2015/01/23 Javascript
面向切面编程(AOP)的理解
2015/05/01 Javascript
javascript检测移动设备横竖屏
2016/05/21 Javascript
canvas 弹幕效果(实例分享)
2017/01/11 Javascript
angular1配合gulp和bower的使用教程
2018/01/19 Javascript
angularjs select 赋值 ng-options配置方法
2018/02/28 Javascript
javascript少儿编程关于返回值的函数内容
2018/05/27 Javascript
详解使用create-react-app添加css modules、sasss和antd
2018/07/31 Javascript
微信小程序开发常见问题及解决方案
2019/07/11 Javascript
js实现无刷新监听URL的变化示例代码详解
2020/06/03 Javascript
[40:31]Secret vs Alliacne 2019国际邀请赛小组赛 BO2 第二场 8.15
2019/08/17 DOTA
python控制台英汉汉英电子词典
2020/04/23 Python
python使用7z解压apk包的方法
2015/04/18 Python
Python中操作文件之write()方法的使用教程
2015/05/25 Python
python实现RSA加密(解密)算法
2016/02/17 Python
python函数中将变量名转换成字符串实例
2020/05/11 Python
python中绕过反爬虫的方法总结
2020/11/25 Python
CSS3 :nth-child()伪类选择器实现奇偶行显示不同样式
2013/11/05 HTML / CSS
ziaja齐叶雅官方海外旗舰店:来自波兰的天然护肤品牌
2017/01/02 全球购物
如何利用find命令查找文件
2016/11/18 面试题
心得体会范文
2014/01/04 职场文书
班班通校本培训方案
2014/03/12 职场文书
幼儿教师寄语集锦
2014/04/03 职场文书
学习雷锋寄语大全
2014/04/11 职场文书
2014年机关党委工作总结
2014/12/11 职场文书
销售员岗位职责
2015/02/10 职场文书
员工工作表扬信
2015/05/05 职场文书
2016年“5.12”护士节慰问信
2015/11/30 职场文书
MySQL主从搭建(多主一从)的实现思路与步骤
2021/05/13 MySQL