Python中的连接符(+、+=)示例详解


Posted in Python onJanuary 13, 2017

前言

本文通过在一段示例代码中发现的问题,来给大家详细介绍了Python中的连接符(+、+=),下面话不多说,来看详细的介绍吧。

假设有下面一段代码:

a = [1, 2, 3, 4]
b = [5, 6, 7, 8, 9]
c = [11, 12, 13, 14, 15, 16, 17, 18, 19, 20]

for item in (a, b, c):
 item += [0] * (10 - len(item))

print a
print b
print c

这段代码的意思是,有三个列表,需要在长度不为 10 的列表尾部填充 0,让其长度变为10。

输出如下:

[1, 2, 3, 4, 0, 0, 0, 0, 0, 0]
[5, 6, 7, 8, 9, 0, 0, 0, 0, 0]
[11, 12, 13, 14, 15, 16, 17, 18, 19, 20]

这里没什么问题,一切正常。但是,现在变了需求,需要在长度不为 10 的列表的前面填充 0。

那么,我们尝试做如下的改动:

a = [1, 2, 3, 4]
b = [5, 6, 7, 8, 9]
c = [11, 12, 13, 14, 15, 16, 17, 18, 19, 20]

for item in (a, b, c):
 item = [0] * (10 - len(item)) + item

print a
print b
print c

直接来看一下输出:

[1, 2, 3, 4]
[5, 6, 7, 8, 9]
[11, 12, 13, 14, 15, 16, 17, 18, 19, 20]

结果却不是我们想象的那样。如果你没有发现问题的所在,就继续往下看吧。当然,如果你已经看出了其中的端倪,那就不需要在这里浪费时间了。

按照我们固有的思维,上面的方法是可行,例如下面的实例:

>>> l = [1, 2, 3, 4, 5]
>>> l = [0]*5 + l
>>> l
[0, 0, 0, 0, 0, 1, 2, 3, 4, 5]

这样的操作让列表如愿以偿的得到我们所期望的改变。

但是,如果我们在其中多加几个步骤呢:

>>> l = [1, 2, 3, 4, 5]
>>> id(l)
139935500860952
>>> l = [0]*5 + l
>>> l
[0, 0, 0, 0, 0, 1, 2, 3, 4, 5]
>>> id(l)
139935500783272

到此,是不是已经看出问题所在了呢。通过 id() 方法的输出可以看到,后边的 “l” 已经不是前边的 “l” 了。

再看看下边的例子:

>>> l = [1, 2, 3, 4, 5]
>>> id(l)
139935500861024
>>> l += [0]*5
>>> l
[1, 2, 3, 4, 5, 0, 0, 0, 0, 0]
>>> id(l)
139935500861024

当用 += 时, “l” 前后是一个。此时,我们应该明白一个事实,文章开头的例子并非莫名其妙,而是有原因的。

别着急,我们再来看看例子:

>>> t = (1, 2, 3, 4, 5)
>>> id(t)
139935501840656
>>> t += (0,)*5
>>> t
(1, 2, 3, 4, 5, 0, 0, 0, 0, 0)
>>> id(t)
139935502151336

可以看到,当我们把列表换成元组时,结果又发生了变化。

那么我们对元组使用 + 操作呢:

>>> t = (1, 2, 3, 4, 5)
>>> id(t)
139935501081200
>>> t = (0,)*5 + t
>>> t
(0, 0, 0, 0, 0, 1, 2, 3, 4, 5)
>>> id(t)
139935502151336

这与列表结果是一样的,没有什么不同。

那么,再来看看字符串呢:

>>> s = "hello"
>>> id(s)
139935500909712
>>> s += "world"
>>> s
'helloworld'
>>> id(s)
139935500909664

结果如同元组,“s” 在使用 += 拼接一个字符串后,被重新赋了值,已然不是之前的变量。反映在内存中就是,“s” 被另外开辟了一个存储空间来存放值。

这里,我们要谈的 Python 连接符就是 + 与 +=。要注意在 Python 中这两个符号有成含义,一个是运用在数学中的加法运算,一个是用在序列类型上的拼接功能。不过,作为加法运算符时,也遵循本文讨论的使用规则。因为讨论这两个符号,本质上是讨论 Python 的 immutable 和 mutable,即可变类型与不可变类型。对可变类型也说,我们可以在原地被变量进行修改,也就是说它的存储空间是可读可写的,例如 list;而对于不可变类型来说,它的存储空间则是只读的,无法对其进行修改,如果需要对不可变类型进行某些操作来得到新的结果,则需要重新开辟一份存储空间来存放这个新产生的结果。

由以上列举的例子,我们可以得到如下的结论:

对于可变类型:

  1. +: 代表连接操作,其结果会创建一个新的对象。
  2. +=: 代表追加操作,即 in-place 操作,在原地把另一个对象的内容追加到对象中。

对于不可变类型: + 与 += 都代表连接或求和操作,两者没有什么区别,其操作的结果都会产生一个新的对象。

下面我们来分析一下文章开头的例子,由于 for 迭代相当于赋值,为了简单起见,我们只分析 a,如下所示:

>>> a = [1, 2, 3, 4]
>>> t = a
>>> id(a)
139712695835400
>>> id(t)
139712695835400
>>> t += [0]*6
>>> t
[1, 2, 3, 4, 0, 0, 0, 0, 0, 0]
>>> id(t)
139712695835400
>>> id(a)
139712695835400
>>> a
[1, 2, 3, 4, 0, 0, 0, 0, 0, 0]
>>>
>>>
>>> a = [1, 2, 3, 4]
>>> t = a
>>> id(a)
139712695835464
>>> id(t)
139712695835464
>>> t = [0]*6 + t
>>> t
[0, 0, 0, 0, 0, 0, 1, 2, 3, 4]
>>> a
[1, 2, 3, 4]
>>> id(a)
139712695835464
>>> id(t)
139712695835400

这里, t 是对 a 的一个引用,就相当于文章开头例子的 item。用 += 对 t 进行操作实际上是对 a 进行操作,而 += 是原地操作,所以改变 t 时,a 也随之变化;如果用 + 对 t 进行操作,在将结果赋值给 t,那么此时的 t 就不再指向 a 了,而是指向 [0]*6 + t,所以 a 没有被改变。

总结

以上就是这篇文章的全部内容了,这里讨论的只是一个简单的问题,而我却用了这么长的篇幅来谈论这个问题,所以我想说的是,对于这些小问题,如果你没有完全理解,那么在程序设计过程中可能会给你带来麻烦。

Python 相关文章推荐
30分钟搭建Python的Flask框架并在上面编写第一个应用
Mar 30 Python
Python序列操作之进阶篇
Dec 08 Python
详解python实现读取邮件数据并下载附件的实例
Aug 03 Python
Python实现运行其他程序的四种方式实例分析
Aug 17 Python
python 时间信息“2018-02-04 18:23:35“ 解析成字典形式的结果代码详解
Apr 19 Python
pyqt5之将textBrowser的内容写入txt文档的方法
Jun 21 Python
利用Python库Scapy解析pcap文件的方法
Jul 23 Python
python实现H2O中的随机森林算法介绍及其项目实战
Aug 29 Python
解决Django删除migrations文件夹中的文件后出现的异常问题
Aug 31 Python
python定时任务 sched模块用法实例
Nov 04 Python
Python+numpy实现矩阵的行列扩展方式
Nov 29 Python
Python selenium 加载并保存QQ群成员,去除其群主、管理员信息的示例代码
May 28 Python
Python中datetime模块参考手册
Jan 13 #Python
python 计算文件的md5值实例
Jan 13 #Python
Python 字典与字符串的互转实例
Jan 13 #Python
python 安装virtualenv和virtualenvwrapper的方法
Jan 13 #Python
Python 使用os.remove删除文件夹时报错的解决方法
Jan 13 #Python
python递归删除指定目录及其所有内容的方法
Jan 13 #Python
Python实现二分查找与bisect模块详解
Jan 13 #Python
You might like
PHP 应用程序的安全 -- 不能违反的四条安全规则
2006/11/26 PHP
php中将字符串转为HTML的实体引用的一个类
2013/02/03 PHP
ThinkPHP 3使用OSS的方法
2018/07/19 PHP
在Ubuntu 18.04上安装PHP 7.3 7.2和7.0的方法
2019/04/09 PHP
怎么用javascript进行拖拽
2006/07/20 Javascript
javascript实现的网页局布刷新效果
2008/12/01 Javascript
jQuery 借助插件Lavalamp实现导航条动态美化效果
2013/09/27 Javascript
nodejs npm包管理的配置方法及常用命令介绍
2014/06/05 NodeJs
JS+CSS实现仿新浪微博搜索框的方法
2015/02/24 Javascript
使用AngularJS和PHP的Laravel实现单页评论的方法
2015/06/19 Javascript
javascript瀑布流式图片懒加载实例
2020/06/28 Javascript
微信小程序 动态的设置图片的高度和宽度详解及实例代码
2017/02/24 Javascript
在node.js中怎么屏蔽掉favicon.ico的请求
2017/03/01 Javascript
vue项目中跳转到外部链接的实例讲解
2018/09/20 Javascript
如何通过shell脚本自动生成vue文件详解
2019/09/10 Javascript
vuex存储token示例
2019/11/11 Javascript
在vue中使用eslint,配合vscode的操作
2020/11/09 Javascript
[48:02]Ti4循环赛第三日 VG vs Liquid和NEWBEE vs DK
2014/07/12 DOTA
[03:59]5分钟带你了解什么是DOTA2(第二期)
2017/02/07 DOTA
Python类的专用方法实例分析
2015/01/09 Python
Python使用itertools模块实现排列组合功能示例
2018/07/02 Python
10个Python小技巧你值得拥有
2018/09/29 Python
python多线程调用exit无法退出的解决方法
2019/02/18 Python
Python List cmp()知识点总结
2019/02/18 Python
python五子棋游戏的设计与实现
2019/06/18 Python
pip安装python库的方法总结
2019/08/02 Python
Michael Kors美国官网:美式奢侈生活风格的代表
2016/11/25 全球购物
日本非常有名的内衣丝袜品牌:GUNZE
2017/01/06 全球购物
Canal官网:巴西女性时尚品牌
2019/10/16 全球购物
澳洲最大的时尚奢侈品电商平台:Cettire
2020/06/15 全球购物
机电一体化职业规划书
2014/01/07 职场文书
竞聘演讲稿
2014/04/24 职场文书
2014年财务工作总结与计划
2014/12/08 职场文书
十佳少年事迹材料
2014/12/25 职场文书
详解Spring事件发布与监听机制
2021/06/30 Java/Android
JavaWeb 入门篇(3)ServletContext 详解 具体应用
2021/07/16 Java/Android