文章来源: Python猫

作者:豌豆花下猫

在 C/C++/Java 等等语言中,整型变量的自增或自减操作是标配,它们又可分为前缀操作(++i 和 --i)与后缀操作(i++ 和 i--),彼此存在着一些细微差别,各有不同的用途。

这些语言的使用者在接触 Python 时,可能会疑惑为什么它不提供 ++ 或 -- 的操作呢?在我前不久发的《Python的十万个为什么?》里,就有不少同学在调查问卷中表示了对此话题感兴趣。

Python 中虽然可能出现 ++i 这种前缀形式的写法,但是它并没有“++”自增操作符,此处只是两个“+”(正数符号)的叠加而已,至于后缀形式的“++”,则完全不支持(SyntaxError: invalid syntax)。

本期“Python为什么”栏目,我们将会从两个主要的角度来回答:Python 为什么不支持 i++ 自增语法?(PS:此处自增指代“自增和自减”,下同)

首先,Python 当然可以实现自增效果,即写成i+=1或者i=i+1,这在其它语言中也是通用的。

虽然 Python 在底层用了不同的魔术方法(__add__()和__iadd__())来完成计算,但表面上的效果完全相同。

所以,我们的问题可以转化成:为什么上面的两种写法会胜过 i++,成为 Python 的最终选择呢?

1、Python 的整数是不可变类型

当我们定义i = 1000时,不同语言会作出不同的处理:

C 之类的语言(写法 int i = 1000)会申请一块内存空间,并给它“绑定”一个固定的名称 i,同时写入一个可变的值 1000。在这里,i 的地址以及类型是固定的,而值是可变的(在一定的表示范围内)Python(写法i = 1000)也会申请一块内存空间,但是它会“绑定”给数字 1000,即这个 1000 的地址以及类型是固定的(immutable),至于 i,只是一个名称标签贴在 1000 上,自身没有固定的地址和类型所以当我们令i“自增”时(i=i+1),它们的处理是不同的:

C 之类的语言先找到 i 的地址上存的数值,然后令它加 1,操作后新的数值就取代了旧的数值Python 的操作过程是把 i 指向的数字加 1,然后把结果绑定到新申请的一块内存空间,再把名称标签 i “贴”到新的数字上。新旧数字可以同时存在,不是取代关系打一个不太恰当的比方:C 中的 i 就像一个宿主,数字 1000 寄生在它上面;而 Python 中的 1000 像个宿主,名称 i 寄生在它上面。C 中的 i 与 Python 中的 1000,它们则寄生在底层的内存空间上……

还可以这样理解:C 中的变量 i 是一等公民,数字 1000 是它的一个可变的属性;Python 中的数字 1000 是一等公民,名称 i 是它的一个可变的属性。

有了以上的铺垫,我们再来看看i++,不难发现:

C 之类的语言,i++ 可以表示 i 的数字属性的增加,它不会开辟新的内存空间,也不会产生新的一等公民Python 之类的语言,i++ 如果是对其名称属性的操作,那样就没有意义了(总不能按字母表顺序,把 i 变成 j 吧);如果理解成对数字本体的操作,那么情况就会变得复杂:它会产生新的一等公民 1001,因此需要给它分配一个内存地址,此时若占用 1000 的地址,则涉及旧对象的回收,那原有对于 1000 的引用关系都会受到影响,所以只能开辟新的内存空间给 1001Python 若支持 i++,其操作过程要比 C 的 i++ 复杂,而且其含义也不再是“令数字增加1”(自增),而是“创建一个新的数字”(新增),这样的话,“自增操作符”(increment operator)就名不副实了。

Python 在理论上可以实现 i++ 操作,但它就必须重新定义“自增操作符”,还会令有其它语言经验的人产生误解,不如就让大家直接写成i += 1或者 i = i + 1好了。

2、Python 有可迭代对象

C/C++ 等语言设计出 i++,最主要的目的是为了方便使用三段式的 for 结构:

for(int i = 0; i < 100; i++){

// 执行 xxx

}

这种程序关心的是数字本身的自增过程,数字做加法与程序体的执行相关联。

Python 中没有这种 for 结构的写法,它提供了更为优雅的方式:

for i in range(100):

# 执行 xxx

my_list = ["你好", "我是Python猫", "欢迎关注"]

for info in my_list:

print(info)

这里体现了不同的思维方式,它关心的是在一个数值范围内的迭代遍历,并不关心也不需要人为对数字做加法。

Python 中的可迭代对象/迭代器/生成器提供了非常良好的迭代/遍历用法,能够做到对 i++ 的完全替代。

例如,上例中实现了对列表内值的遍历,Python 还可以用 enumerate() 实现对下标与具体值的同时遍历:

my_list = ["你好", "我是Python猫", "欢迎关注"]

for i, info in enumerate(my_list):

print(i, info)

# 打印结果:

0 你好

1 我是Python猫

2 欢迎关注

再例如对于字典的遍历,Python 提供了 keys()、values()、items() 等遍历方法,非常好用:

my_dict = {'a': '1', 'b': '2', 'c': '3'}

for key in my_dict.keys():

print(key)

for key, value in my_dict.items():

print(key, value)

有了这样的利器,哪里还有 i++ 的用武之地呢?

不仅如此,Python 中基本上很少使用i += 1或者 i = i + 1,由于存在着随处可见的可迭代对象,开发者们很容易实现对一个数值区间的操作,也就很少有对于某个数值作累加的诉求了。

所以,回到我们开头的问题,其实这两种“自增”写法并没有胜出 i++ 多少,只因为它们是通用型操作,又不需要引入新的操作符,所以 Python 才延续了一种基础性的支持。真正的赢家其实是各种各样的可迭代对象!

稍微小结下:Python 不支持自增操作符,一方面是因为它的整数是不可变类型的一等公民,自增操作(++)若要支持,则会带来歧义;另一方面主要因为它有更合适的实现,即可迭代对象,对遍历操作有很好的支持。

python自增_Python 为什么不支持 i++ 自增语法,不提供 ++ 操作符?相关推荐

  1. python为什么不能自动语法_Python 为什么不支持 i++ 自增语法,不提供 ++ 操作符?...

    在 C/C++/Java 等等语言中,整型变量的自增或自减操作是标配,它们又可分为前缀操作(++i 和 --i)与后缀操作(i++ 和 i--),彼此存在着一些细微差别,各有不同的用途. 这些语言的使 ...

  2. c++提供的可有效分配对象空间的运算符是_Python 为什么不支持 i++ 自增语法,不提供 ++ 操作符?

    在 C/C++/Java 等等语言中,整型变量的自增或自减操作是标配,它们又可分为前缀操作(++i 和 --i)与后缀操作(i++ 和 i--),彼此存在着一些细微差别,各有不同的用途. 这些语言的使 ...

  3. python数据库教程_Python连接mysql数据库及简单增删改查操作示例代码

    1.安装pymysql 进入cmd,输入 pip install pymysql: 2.数据库建表 在数据库中,建立一个简单的表,如图: 3.简单操作 3.1查询操作 #coding=utf-8 #连 ...

  4. python pymysql实例_Python使用pymysql模块操作mysql增删改查实例分析

    Python使用pymysql模块操作mysql增删改查实例分析 发布时间:2020-09-30 16:42:12 来源:脚本之家 阅读:92 本文实例讲述了Python使用pymysql模块操作My ...

  5. python bytes 操作_Python 3:不支持/:“float”和“bytes”的操作数类型

    我正在缓慢地尝试将一个旧的python 2代码转换为python 3,但是我遇到了一个障碍,代码如下:seconds = functools.reduce(lambda x,y:x*60+y,map( ...

  6. python xpath爬虫_Python爬虫(2):XPath语法

    OK,上次我们说到了网页爬虫的一个思路:首先说一下网页爬虫的整个思路方法:先爬取整个网页,也就是将网页的源代码给获取下来 爬取下来的网页再通过文本解析提取,找到我们需要的信息,可以是图片或者文字 然后 ...

  7. Python 为什么不支持 i++ 自增语法,不提供 ++ 操作符?

    点击上方"小白学视觉",选择加"星标"或"置顶" 重磅干货,第一时间送达 在 C/C++/Java 等等语言中,整型变量的自增或自减操作是标 ...

  8. Python 为什么不支持 i++ 自增语法,不提供 ++ 操作符

    在 C/C++/Java 等等语言中,整型变量的自增或自减操作是标配,它们又可分为前缀操作(++i 和 --i)与后缀操作(i++ 和 i--),彼此存在着一些细微差别,各有不同的用途. 这些语言的使 ...

  9. python真值是什么意思_Python 为什么能支持任意的真值判断?

    本文出自"Python为什么"系列,请查看全部文章 Python 在涉及真值判断(Truth Value Testing)时,语法很简便. 比如,在判断某个对象是否不为 None ...

最新文章

  1. 浅谈WebSocket
  2. python无师自通配套资源_Python Tkinter Pack布局管理器(超级详细,看了无师自通)...
  3. SET-UID程序漏洞实验
  4. [Unity动画]04.Avatar Mask
  5. 重做53. Maximum Subarray
  6. LVS+DR源码安装
  7. 今日代码(200727)--全局空间自相关性
  8. Linux 进程、父进程、子进程
  9. 如何修改计算机无线mac地址,修改计算机mac地址_怎么修改mac地址
  10. codeforce438D The Child and Sequence
  11. 多媒体实时交互系统主要由系统服务器,多媒体设备和多媒体交互系统专利_专利申请于2017-03-08_专利查询 - 天眼查...
  12. Linux 自定义快捷命令
  13. 死亡之ping (ping of death)
  14. 总结卡方检验(Chi-square test)和费舍尔精确检验(Fisher exact test)的区别
  15. docker英文文档和中文文档
  16. Java Web课程设计
  17. 小程序实现图片双滑放大缩小
  18. 企业运维之七层负载均衡--Haproxy
  19. 对校招生培养工作的建议_学校招生工作的几点建议
  20. Python基础笔记_Day01_计算机基础知识和Python开发环境搭建

热门文章

  1. 总结常见的ES6新语法特性
  2. [Vue.js]实战 -- 电商项目(三)
  3. MySQL_自定义函数
  4. LeetCode--5.最长回文子串(滑动窗口)
  5. 7-34 红色警报 (10 分)(结构体并查集)
  6. mantis apache mysql_Debian下mysql5+php5+apache2+mantis服务器升级
  7. mkdir: 无法创建目录“/home/lj/.tldr“: 文件已存在
  8. 用c语言计算2的n次方,计算2的N次方........有什么错吗?
  9. java中j 和 j啥区别_从字节码层次分析++j和j++的区别
  10. 2.privite私有变量的意义