Python 序列解包中的tricky point

Python 序列解包中的tricky point

奇怪的现象

先来看一个违反直觉的程序

a = [1,2,3,4,5,6,7,8,9]
b = a.copy()
a[:-3], a[-3:] = a[-3:],a[:-3]
b[:6], b[6:] = b[6:],b[:6]

这里乍看起来a和b的运行结果应该是一样的,可是实际运行程序却得到了这样的结果

在这里插入图片描述

简化问题

为了细探发生了什么,我决定分解这个式子,一步一步地进行。

在这里插入图片描述

然而这次竟然又得到了与刚刚不同的结果。不过仔细观察之下原因已经十分清楚了,本次的程序和前一个程序的区别在于赋值顺序的不同。

在给普通元素赋值时,顺序不同可能没有什么明显的后果,但是在这个例子中,不同的顺序会导致完全不同的结果,这是因为在进行第一次赋值以后,列表的长度已经发生了改变,新的索引会以改变之后的列表为准,而不是我一开始直觉中认为的还是原始列表索引的位置。

这样看来,上面那个奇怪的例子就非常容易理解了。这里我们以这样一个程序来彰显这种显著的区别:

在这里插入图片描述

可以看到a,b两个列表的赋值仅有顺序上的区别,但结果是完全不同的,这提示我们在进行这样的切片和序列解包的结合时,要非常注意顺序的问题。

总结

那么经过上面的思考,我们是否可以总结出一些经验以避免出错呢?

我的思考是:既然与直觉不符是因为列表长度改变导致索引变化,那我们要做的就是让第一个赋值进行后,尽量不影响第二个赋值的索引位置。以本例中通过切片交换为例:

  • 当我们使用正数索引时,因为序列计数是从列表开头开始的,所以应该先赋值末尾的元素,再赋值开头的元素,这样不会互相干扰。
  • 当我们使用负数索引时,因为序列计数是从列表末尾开始的,所以应该先赋值开头的元素,再赋值末尾的元素,这样不会互相干扰。

即对应的赋值语句应写为如下的形式:

  • 正数:a[index:], a[:index] = a[:index],a[index:]
  • 负数:a[:-index], a[-index:] = a[-index:],a[:-index]

我们来试验一下:

在这里插入图片描述

可以看到,a和b行为变得一致了。

这时我们再回头看一开始提出的问题就非常显然了,因为正数索引的赋值顺序写反了,自然得不到预期的结果。至此,这个问题完美解决。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值