一文带你了解Python中的双下方法

 更新时间:2022年07月14日 10:51:08   作者:程序员小六  
Python中有一些特殊方法的方法名都是以双下划线开始和结束,所以又被称为双下方法。本文就来为大家详细讲讲Python中的双下方法的使用,感兴趣的可以了解一下
(福利推荐:【腾讯云】服务器最新限时优惠活动,云服务器1核2G仅99元/年、2核4G仅768元/3年,立即抢购>>>:9i0i.cn/qcloud

(福利推荐:你还在原价购买阿里云服务器?现在阿里云0.8折限时抢购活动来啦!4核8G企业云服务器仅2998元/3年,立即抢购>>>:9i0i.cn/aliyun

前言

大家在写 Python 代码的时候有没有这样的疑问。

为什么数学中的+号,在字符串运算中却变成拼接功能,如'ab' + 'cd'结果为abcd;而*号变成了重复功能,如'ab' * 2结果为abab

为什么某些对象print能输出数据,而print自定义的类对象却输出一堆看不懂的代码<__main__.MyCls object at 0x105732250>

不是因为系统做了特殊定制,而是 Python 中有一类特殊的方法,在某些特定的场合会自动调用。如,在字符串类str中定义了__add__方法后,当代码遇到字符串相加'ab' + 'cd'时,就会自动调用__add__方法完成字符串拼接。

因为这类特殊方法的方法名都是以双下划线开始和结束,所以又被称为双下方法。

Python 中的双下方法很多,今天我们对它做个详解。

Python中的双下方法

1. init方法

__init__的方法是很多人接触的第一个双下方法

class?A:
????def?__init__(self,?a):
????????self.a?=?a

当调用A()实例化对象的时候,__init__方法会被自动调用,完成对象的初始化。

2. 运算符的双下方法

在类中定义运算符相关的双下方法,可以直接在类对象上做加减乘除、比较等操作。

这里,定义一个尺子类Rule,它包含一个属性r_len代表尺子的长度。

class?Rule:
????def?__init__(self,?r_len):
????????self.r_len?=?r_len

2.1 比较运算符

如果想按照尺子的长度对不同的尺子做比较,需要在Rule类中定义比较运算符。

class?Rule:
????def?__init__(self,?r_len):
????????self.r_len?=?r_len

????#?<?运算符
????def?__lt__(self,?other):
????????return?self.r_len?<?other.r_len

????#?<=?运算符
????def?__le__(self,?other):
????????return?self.r_len?<=?other.r_len

????#?>?运算符
????def?__gt__(self,?other):
????????return?self.r_len?>?other.r_len

????#?>=?运算符
????def?__ge__(self,?other):
????????return?self.r_len?>=?other.r_len

这里定义了<<=>>=四个比较运算符,这样就可以用下面的代码比较Rule对象了。

rule1?=?Rule(10)
rule2?=?Rule(5)
print(rule1?>?rule2)??#?True
print(rule1?>=?rule2)??#?True
print(rule1?<?rule2)??#?False
print(rule1?<=?rule2)??#?False

当用>比较rule1rule2的时候,rule1对象会自动调用__gt__方法,并将rule2对象传给other参数,完成比较。

下面是比较运算符的双下方法

比较运算符双下方法

2.2 算术运算符

可以支持类对象加减乘除。

def?__add__(self,?other):
????return?Rule(self.r_len?+?other.r_len)

这里定义了__add__方法,对应的是+运算符,他会把两个尺子的长度相加,并生成新的尺子。

rule1?=?Rule(10)
rule2?=?Rule(5)
rule3?=?rule1?+?rule2

下面是算术运算符的双下方法

2.3 反向算术运算符

它支持其他类型的变量与Rule类相加。以__radd__方法为例

def?__radd__(self,?other):
????return?self.r_len?+?other
rule1?=?Rule(10)
rule2?=?10?+?rule1

程序执行10 + rule1时,会尝试调用int类的__add__int类类没有定义与Rule类对象相加的方法,所以程序会调用+号右边对象rule1__radd__方法,并把10传给other参数。

所以这种运算符又叫右加运算符。它所支持的运算符与上面的算术运算符一样,方法名前加r即可。

2.4 增量赋值运算符

增量赋值运算符是+=-=*=/=等。

def?__iadd__(self,?other):
????self.r_len?+=?other
????return?self
rule1?=?Rule(10)
rule1?+=?5

除了__divmod__方法,其他的跟算数运算符一样,方面名前都加i。

2.4 位运算符

这部分支持按二进制进行取反、移位和与或非等运算。由于Rule类不涉及位运算,所以我们换一个例子。

定义二进制字符串的类BinStr,包含bin_str属性,表示二进制字符串。

class?BinStr:
????def?__init__(self,?bin_str):
????????self.bin_str?=?bin_str
x?=?BinStr('1010')??#创建二进制字符串对象
print(x.bin_str)?#?1010

BinStr定义一个取反运算符~

#?~?运算符
def?__invert__(self):
????inverted_bin_str?=?''.join(['1'?if?i?==?'0'?else?'0'?for?i?in?self.bin_str])
????return?BinStr(inverted_bin_str)

__invert__方法中,遍历bin_str字符串,将每位取反,并返回一个新的BinStr类对象。

x?=?BinStr('1011')

invert_x?=?~x
print(invert_x.bin_str)?#?0100

下面是位运算符的双下方法

这部分也支持反向位运算符和增量赋值位运算符,规则跟算数运算符一样,这里就不再赘述。

3.字符串表示

这部分涉及两个双下方法__repr____format__,在某些特殊场景,如print,会自动调用,将对象转成字符串。

还是以BinStr为例,先写__repr__方法。

def?__repr__(self):
????decimal?=?int('0b'+self.bin_str,?2)
????return?f'二进制字符串:{self.bin_str},对应的十进制数字:{decimal}'
x?=?BinStr('1011')
print(x)
#?输出:二进制字符串:1011,对应的十进制数字:11

当程序执行print(x)时,会自动调用__repr__方法,获取对象x对应的字符串。

再写__format__方法,它也是将对象格式化为字符串。

def?__format__(self,?format_spec):
????return?format_spec?%?self.bin_str
print('{0:二进制字符串:%s}'.format(x))
#?输出:二进制字符串:1011

.format方法的前面字符串里包含0:时,就会自动调用__format__方法,并将字符串传给format_spec参数。

4.数值转换

调用int(obj)float(obj)等方法,可以将对象转成相对应数据类型的数据。

def?__int__(self):
????return?int('0b'+self.bin_str,?2)
x?=?BinStr('1011')
print(int(x))

当调用int(x)时,会自动调用__int__方法,将二进制字符串转成十进制数字。

数值转换除了上面的两个外,还有__abs____bool____complex____hash____index____str__

__str____repr__一样,在print时都会被自动调用,但__str__优先级更高。

5.集合相关的双下方法

这部分可以像集合那样,定义对象长度、获取某个位置元素、切片等方法。

__len____getitem__为例

def?__len__(self):
????return?len(self.bin_str)

def?__getitem__(self,?item):
????return?self.bin_str[item]
x?=?BinStr('1011')

print(len(x))??#?4
print(x[0])??#?1
print(x[0:3])??#?101

len(x)会自动调用__len__返回对象的长度。

通过[]方式获取对象的元素时,会自动调用__getitem__方法,并将切片对象传给item参数,即可以获取单个元素,还可以获取切片。

集合相关的双下方法还包括__setitem____delitem____contains__

6.迭代相关的双下方法

可以在对象上使用for-in遍历。

def?__iter__(self):
????self.cur_i?=?-1
????return?self

def?__next__(self):
????self.cur_i?+=?1
????if?self.cur_i?>=?len(self.bin_str):
????????raise?StopIteration()??#?退出迭代
????return?self.bin_str[self.cur_i]
x?=?BinStr('1011')
for?i?in?x:
????print(i)

当在x上使用for-in循环时,会先调用__iter__方法将游标cur_i置为初始值-1,然后不断调用__next__方法遍历self.bin_str中的每一位。

这部分还有一个__reversed__方法用来反转对象。

def?__reversed__(self):
????return?BinStr(''.join(list(reversed(self.bin_str))))
x?=?BinStr('1011')
reversed_x?=?reversed(x)
print(reversed_x)
#?输出:二进制字符串:1101,对应的十进制数字:13

7.类相关的双下方法

做 web 开发的朋友,用类相关的双下方法会更多一些。

7.1 实例的创建和销毁

实例的创建是__new____init__方法,实例的销毁是__del__方法。

__new__的调用早于__init__,它的作用是创建对象的实例(内存开辟一段空间),而后才将该实例传给__init__方法,完成实例的初始化。

由于__new__是类静态方法,因此它可以控制对象的创建,从而实现单例模式

__del__方法在实例销毁时,被自动调用,可以用来做一些清理工作和资源释放的工作。

7.2 属性管理

类属性的访问和设置。包括__getattr____getattribute____setattr____delattr__方法。

__getattr____getattribute__的区别是,当访问类属性时,无论属性存不存在都会调用__getattribute__方法,只有当属性不存在时才会调用__getattr__方法。

7.3 属性描述符

控制属性的访问,一般用于把属性的取值控制在合理范围内。包括__get____set____delete__方法。

class?XValidation:
????def?__get__(self,?instance,?owner):
????????return?self.x

????def?__set__(self,?instance,?value):
????????if?0?<=?value?<=?100:
????????????self.x?=?value
????????else:
????????????raise?Exception('x不能小于0,不能大于100')

????def?__delete__(self,?instance):
????????print('删除属性')


class?MyCls:
????x?=?XValidation()

????def?__init__(self,?n):
????????self.x?=?n

obj?=?MyCls(10)
obj.x?=?101
print(obj.x)?#?抛异常:Exception: x不能小于0,不能大于100

上述例子,通过类属性描述符,可以将属性x的取值控制在[0, 100]之前,防止不合法的取值。

8.总结

虽然上面介绍的不是所有的双下方法,但也算是绝大多数了。

虽然双下方法里可以编写任意代码,但大家尽量编写与方法要求一样的代码。如,在__add__方法实现的不是对象相加而是相减,虽然也能运行,但这样会造成很大困惑,不利于代码维护。

到此这篇关于一文带你了解Python中的双下方法的文章就介绍到这了,更多相关Python双下方法内容请搜索程序员之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持程序员之家!

相关文章

  • 基于python中theano库的线性回归

    基于python中theano库的线性回归

    这篇文章主要为大家详细介绍了基于python中theano库的线性回归,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-08-08
  • wxPython框架类和面板类的使用实例

    wxPython框架类和面板类的使用实例

    这篇文章主要介绍了wxPython框架类和面板类的使用实例,主要实现了自定义框架类及其完整的调用方法,代码简单高效,是进一步学习wxPython框架类的基础,需要的朋友可以参考下
    2014-09-09
  • numpy给array增加维度np.newaxis的实例

    numpy给array增加维度np.newaxis的实例

    今天小编就为大家分享一篇numpy给array增加维度np.newaxis的实例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-11-11
  • python利用proxybroker构建爬虫免费IP代理池的实现

    python利用proxybroker构建爬虫免费IP代理池的实现

    这篇文章主要介绍了python利用proxybroker构建爬虫免费IP代理池,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-02-02
  • Python光学仿真wxpython透镜演示系统框架

    Python光学仿真wxpython透镜演示系统框架

    这篇文章主要为大家介绍了Python光学仿真UI界面的wxpython透镜演示系统框架基本讲解,有需要的朋友可以借鉴参考下,希望能够有所帮助
    2021-10-10
  • python对一个数向上取整的实例方法

    python对一个数向上取整的实例方法

    在本篇文章中小编给大家整理了关于python对一个数向上取整的实例方法,需要的朋友们可以跟着学习下。
    2020-06-06
  • Python中win32com模块的使用

    Python中win32com模块的使用

    本文主要介绍了Python中win32com模块的使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-01-01
  • Python递归实现汉诺塔算法示例

    Python递归实现汉诺塔算法示例

    这篇文章主要介绍了Python递归实现汉诺塔算法,结合实例形式分析了汉诺塔算法的原理、实现步骤与相关操作技巧,需要的朋友可以参考下
    2018-03-03
  • python3爬虫中异步协程的用法

    python3爬虫中异步协程的用法

    在本篇文章里小编给大家整理的是关于python3爬虫中异步协程的用法,需要的朋友们可以学习参考下。
    2020-07-07
  • Pyinstaller加密打包应用的示例代码

    Pyinstaller加密打包应用的示例代码

    这篇文章主要介绍了Pyinstaller加密打包应用的示例代码,代码简单易懂,非常不错,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-06-06

最新评论

?


http://www.vxiaotou.com