鱼C论坛

 找回密码
 立即注册
查看: 1966|回复: 2

[技术交流] python3中变量赋值问题分析

[复制链接]
发表于 2017-7-9 17:50:36 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

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
>>>

本帖被以下淘专辑推荐:

想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2017-7-9 18:37:43 | 显示全部楼层
even = odd = [] 是一个数组,even = [], odd = [] 则是两个数组

在 Python 中,变量名都是标签。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2017-7-10 15:27:16 | 显示全部楼层
秒懂,哈哈,不过楼主很耐心细致,解释了这么一大通
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2024-11-22 19:54

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表