疑惑在于 《python 程序设计(第2版) 董付国 清华大学出版社》第46页
原文是:
【列表推导式】 >>>freshfruit=['banana', 'loganberry', 'passion fruit'] >>>aList=[w.strip() for w in freshfruit] 等价于下面的代码: >>>freshfruit=['banana', 'loganberry', 'passion fruit'] >>>for i, v in enumerate(freshfruit): freshfruit[i]=v.strip() 同时,也等价于 >>>freshfruit=['banana', 'loganberry', 'passion fruit'] >>>freshfruit=list(map(str.strip, freshfruit))
关键疑惑点在于最后一行代码中的 map(str.strip, freshfruit)
根据上一条随笔, map(A, B)函数是 把 B 代入 A 中 返回结果,按这个来看,那么原文应该是 把列表 freshfruit 中的元素代入 str.strip 中,可是 str.strip 是什么呢?!
先来看看这些代码输出的结果是什么:
['banana', 'loganberry', 'passion fruit']
结果竟然与 原来的列表(freshfruit) 内容一样
判断 id(freshfruit) == id(aList) 答案是 False
故可知道这些代码作用是 列表内容的复制,而不是内存地址的复制,即不是 ‘is’ 的关系,而是 ‘==’ 的关系。
到这里就能明白 列表的复制 不能是 A = B 这么简单,因为这样就是内存地址的相等,改变其中一个列表就会同样对另一个列表进行改变。
比如说:
>>> oldlist = ['a', 'b', 'c'] >>> newlist = oldlist >>> oldlist ['a', 'b', 'c'] >>> newlist ['a', 'b', 'c'] >>> newlist.append('d') >>> newlist ['a', 'b', 'c', 'd'] >>> oldlist ['a', 'b', 'c', 'd']
改变了 newlist 也会对 oldlist 进行改变。
>>> id(oldlist) 1625304 >>> id(newlist) 1625304 >>> newlist is newlist True
内存地址一样。
那么回到原来的疑惑, str.strip 到底是什么?
到这里我才忽然明白了, 根据 strip()函数 的意思, Python strip() 方法用于移除字符串头尾指定的字符(默认为空格或换行符)或字符序列。——by 菜鸟教程
而书本上的那段代码其实是这样的:
>>>freshfruit=[' banana ', ' loganberry ', 'passion fruit ']
在字符前后是加了几个空格的,而不是像我最开始敲的代码一样没有空格。
★这就解释了其实书本上代码的意思是 去除空格 ,而不是我以为的 列表内容的复制★
疑惑解决!
另附上 列表推导式 的常用方法(来源书本):
①实现嵌套列表的平铺
>>>vec=[[1,2,3], [4,5,6], [7,8,9]] >>>[num for elem in vec for num in elem] 结果是: [1,2,3,4,5,6,7,8,9]
②列出当前文件夹下所有Python源文件(以 .py 结尾的文件)
>>>import os >>>[filename for filename in os.listdir('.') if filename.endwith('.py')]
③实现矩阵转置
>>>matrix=[[1,2,3,4], [5,6,7,8], [9,10,11,12]] >>>[[row[i] for row in matrix] for i in range(4)] 结果是: [[1,5,9], [2,6,10], [3,7,11], [4,8,12]]
也可以用 zip() 和 list() 实现矩阵转置
>>>list(zip(*matrix))
④文件对象迭代
>>>fp=open('D:/XXX.txt', 'r') >>>print([line for line in fp])
⑤生成100以内的所有素数
>>>[p for p in range(2, 100) if 0 not in [p%d for d in range(2, int(sqrt(p)) +1)]]
⑥求小于10000的同构数(这个是我以前在CSDN上写的)
from math import ceil print([i for i in range(1,10000) for n in range(len(str(i * i)),len(str(i * i)) + 1) if i == i * i % (10**(ceil(n / 2))) ])