Python深入学习之特殊方法与多范式


Posted in Python onAugust 31, 2014

Python一切皆对象,但同时,Python还是一个多范式语言(multi-paradigm),你不仅可以使用面向对象的方式来编写程序,还可以用面向过程的方式来编写相同功能的程序(还有函数式、声明式等,我们暂不深入)。Python的多范式依赖于Python对象中的特殊方法(special method)。

特殊方法名的前后各有两个下划线。特殊方法又被成为魔法方法(magic method),定义了许多Python语法和表达方式,正如我们在下面的例子中将要看到的。当对象中定义了特殊方法的时候,Python也会对它们有“特殊优待”。比如定义了__init__()方法的类,会在创建对象的时候自动执行__init__()方法中的操作。

(可以通过dir()来查看对象所拥有的特殊方法,比如dir(1))

运算符

Python的运算符是通过调用对象的特殊方法实现的。比如:

'abc' + 'xyz'               # 连接字符串

实际执行了如下操作:
'abc'.__add__('xyz')

所以,在Python中,两个对象是否能进行加法运算,首先就要看相应的对象是否有__add__()方法。一旦相应的对象有__add__()方法,即使这个对象从数学上不可加,我们都可以用加法的形式,来表达obj.__add__()所定义的操作。在Python中,运算符起到简化书写的功能,但它依靠特殊方法实现。

Python不强制用户使用面向对象的编程方法。用户可以选择自己喜欢的使用方式(比如选择使用+符号,还是使用更加面向对象的__add__()方法)。特殊方法写起来总是要更费事一点。

尝试下面的操作,看看效果,再想想它的对应运算符

(1.8).__mul__(2.0)
True.__or__(False)

内置函数

与运算符类似,许多内置函数也都是调用对象的特殊方法。比如

len([1,2,3])      # 返回表中元素的总数

实际上做的是
[1,2,3].__len__()

相对与__len__(),内置函数len()也起到了简化书写的作用。

尝试下面的操作,想一下它的对应内置函数

(-1).__abs__()
(2.3).__int__()

表(list)元素引用

下面是我们常见的表元素引用方式

li = [1, 2, 3, 4, 5, 6]

print(li[3])

上面的程序运行到li[3]的时候,Python发现并理解[]符号,然后调用__getitem__()方法。
li = [1, 2, 3, 4, 5, 6]

print(li.__getitem__(3))

尝试看下面的操作,想想它的对应
li.__setitem__(3, 0)
{'a':1, 'b':2}.__delitem__('a')

函数

我们已经说过,在Python中,函数也是一种对象。实际上,任何一个有__call__()特殊方法的对象都被当作是函数。比如下面的例子:

class SampleMore(object):

    def __call__(self, a):

        return a + 5
add = SampleMore()     # A function object

print(add(2))          # Call function    

map(add, [2, 4, 5])    # Pass around function object

add为SampleMore类的一个对象,当被调用时,add执行加5的操作。add还可以作为函数对象,被传递给map()函数。

当然,我们还可以使用更“优美”的方式,想想是什么。

总结

对于内置的对象来说(比如整数、表、字符串等),它们所需要的特殊方法都已经在Python中准备好了。而用户自己定义的对象也可以通过增加特殊方法,来实现自定义的语法。特殊方法比较靠近Python的底层,许多Python功能的实现都要依赖于特殊方法。我们将在以后看到更多的例子。

Python的许多语法都是基于其面向对象模型的封装。对象模型是Python的骨架,是功能完备、火力强大的大黄蜂。但是Python也提供更加简洁的语法,让你使用不同的编程形态,从而在必要时隐藏一些面向对象的接口。正如我们看到的Camaro跑车,将自己威风的火药库收起来,提供方便人类使用的车门和座椅。

Python 相关文章推荐
Python 列表排序方法reverse、sort、sorted详解
Jan 22 Python
python3+mysql查询数据并通过邮件群发excel附件
Feb 24 Python
python实现12306抢票及自动邮件发送提醒付款功能
Mar 08 Python
使用11行Python代码盗取了室友的U盘内容
Oct 23 Python
Python异常的检测和处理方法
Oct 26 Python
使用python将请求的requests headers参数格式化方法
Jan 02 Python
Python中如何导入类示例详解
Apr 17 Python
使用python判断jpeg图片的完整性实例
Jun 10 Python
Pytorch Tensor的索引与切片例子
Aug 18 Python
keras 获取某层的输入/输出 tensor 尺寸操作
Jun 10 Python
Python字符串查找基本操作代码案例
Oct 27 Python
python 检测nginx服务邮件报警的脚本
Dec 31 Python
python中的reduce内建函数使用方法指南
Aug 31 #Python
Python中使用ConfigParser解析ini配置文件实例
Aug 30 #Python
python进阶教程之动态类型详解
Aug 30 #Python
python进阶教程之异常处理
Aug 30 #Python
python进阶教程之函数对象(函数也是对象)
Aug 30 #Python
python进阶教程之循环对象
Aug 30 #Python
python进阶教程之循环相关函数range、enumerate、zip
Aug 30 #Python
You might like
PHP 简易输出CSV表格文件的方法详解
2013/06/20 PHP
PHP和javascript常用正则表达式及用法实例
2014/07/01 PHP
ThinkPHP5&5.1框架关联模型分页操作示例
2019/08/03 PHP
限制复选框的最大可选数
2006/07/01 Javascript
img标签中onerror用法
2009/08/13 Javascript
js判断浏览器类型为ie6时不执行
2014/06/15 Javascript
js实现滚动条滚动到某个位置便自动定位某个tr
2021/01/20 Javascript
jQuery+CSS实现一个侧滑导航菜单代码
2016/05/09 Javascript
jQuery获取attr()与prop()属性值的方法及区别介绍
2016/07/06 Javascript
jquery datatable服务端分页
2016/08/31 Javascript
利用Node.js了解与测量HTTP所花费的时间详解
2017/09/22 Javascript
[01:53]3.19 DOTA2发布会 现场精彩Coser表演
2014/03/25 DOTA
Python中声明只包含一个元素的元组数据方法
2014/08/25 Python
Python实现配置文件备份的方法
2015/07/30 Python
Python的Flask开发框架简单上手笔记
2015/11/16 Python
用Python解决计数原理问题的方法
2016/08/04 Python
python matplotlib 注释文本箭头简单代码示例
2018/01/08 Python
对pyqt5中QTabWidget的相关操作详解
2019/06/21 Python
python requests指定出口ip的例子
2019/07/25 Python
Django Admin后台添加数据库视图过程解析
2020/04/01 Python
Visual Studio Code搭建django项目的方法步骤
2020/09/17 Python
IRO美国官网:法国服装品牌
2018/03/06 全球购物
Laravel的加密解密与哈希实例讲解
2021/03/24 PHP
创业计划书如何吸引他人眼球
2014/01/10 职场文书
小学新学期教师寄语
2014/01/18 职场文书
社区端午节活动方案
2014/01/28 职场文书
秋季运动会广播稿大全
2014/02/17 职场文书
综艺节目策划方案
2014/06/13 职场文书
个人授权委托书范文
2014/09/21 职场文书
2015清明节祭奠英烈寄语大全
2015/03/04 职场文书
暑假打工感想
2015/08/07 职场文书
HR在给员工开具离职证明时,需要注意哪些问题?
2019/07/03 职场文书
SqlServer 垂直分表(减少程序改动)
2021/04/16 SQL Server
Python 循环读取数据内存不足的解决方案
2021/05/25 Python
vue使用wavesurfer.js解决音频可视化播放问题
2022/04/04 Vue.js
小程序自定义轮播图圆点组件
2022/06/25 Javascript