|
马上注册,结交更多好友,享用更多功能^_^
您需要 登录 才可以下载或查看,没有账号?立即注册
x
本帖最后由 chunchun2017 于 2017-7-9 17:53 编辑
今天在编写一个小程序的时候,遇到一个有趣的现象,过程如下:
要编写的小程序实现的效果:
1. 程序运行后,提示输入一列数组,以逗号分隔,然后回车
2. 回车后,程序自动将步骤1中输入的所有奇数放入一个列表,将步骤1中输入的所有偶数放入另一个列表。
程序代码如下:
====================================================
#自动将输入的一组数分成奇偶数列,并打印出来
even=odd=[]
list0=input('请输入一组数,以英文输入法的逗号分隔....\n').split(',')
while(len(list0)>0):
number=int(list0.pop())
if(number%2==0):
even.append(number)
else:
odd.append(number)
print('您输入的数列中,偶数有:',even)
print('您输入的数列中,奇数有:',odd)
========================================================
尼玛!没想到运行结果如下:
请输入一组数,以英文输入法的逗号分隔....
10,23,90,45,988,134,108,435,20
您输入的数列中,偶数有: [20, 435, 108, 134, 988, 45, 90, 23, 10]
您输入的数列中,奇数有: [20, 435, 108, 134, 988, 45, 90, 23, 10]
>>>
这是神马情况?偶数,奇数列表一模一样,都是原数组的内容?程序代码的逻辑是木有错误的啊!
经过分析后,将代码做了微小的调整,调整后的代码如下:
============================================
#自动将输入的一组数分成奇偶数列,并打印出来
even=[]
odd=[] #仅仅将even=odd=[]改成了even=[],odd=[],后面的代码没有修改
list0=input('请输入一组数,以逗号分隔....\n').split(',')
while(len(list0)>0):
number=int(list0.pop())
if(number%2==0):
even.append(number)
else:
odd.append(number)
print('您输入的数列中,偶数有:',even)
print('您输入的数列中,奇数有:',odd)
==================================
运行结果如下:
请输入一组数,以逗号分隔....
10,20,33,45,86,981,100
您输入的数列中,偶数有: [100, 86, 20, 10]
您输入的数列中,奇数有: [981, 45, 33]
>>> 正确!
================================================================
为什么even=odd=[]就不行,而even=[],odd=[]就可以呢?
原来写成even=odd=[],python会让even,odd变量指向同一个内存地址,两个变量指向同一个地址,这样奇数也会存到这个地址对应的存储空间,偶数也会存到这个地址对应的存储空间,最终结果当然是两个变量都显示同样的内容,而且是输入的数组的全部内容!而even=[],odd=[],则会分别为even和odd各自分配一个内存地址。
看来Python虽然崇尚简洁,但是有时候不能盲目简洁啊!
====================
在网上找到了这样一段话:
python中会为每个出现的对象分配内存,哪怕他们的值完全相等(注意是相等不是相同)。如执行a=2.0,b=2.0这两个语句时会先后为2.0这个Float类型对象分配内存,然后将a与b分别指向这两个对象。所以a与b指向的不是同一对象:
a=2.0
b=2.0
a is b # 结果为False
a == b # 结果为True
但是为了提高内存利用效率对于一些简单的对象,如一些数值较小的int对象,python采取重用对象内存的办法,如指向a=2,b=2时,由于2作为简单的int类型且数值小,python不会两次为其分配内存,而是只分配一次,然后将a与b同时指向已分配的对象:
a=2
b=2
a is b # 结果为True
a == b # 结果为True
如但果赋值的不是2而是大的数值,情况就跟前面的一样了:
a=4444
b=4444
a is b # 结果为False
a == b # 结果为True
这里is 和==类似编译原理中传值于传地址。又或者说是is只是传递的指针,判断是否指向同一个地址块,这样is两边的参数指向内存中同个地址块,毕竟我家电视跟你电视不是同一个东西。而==则是仅仅判断值相同
================================================
通过测试,发现上面的话是正确的:
>>> a=b=[]
>>> a is b
True
>>> a == b
True
>>> a=[]
>>> b=[]
>>> a is b
False
>>> a == b
True
>>>
|
|