早上看一本Python的书上写了这么一句话:
for循环是一种遍历列表的有效方式,但在for循环中不应修改列表,否则将导致Python难以跟踪其中的元素
其大意指的就是如果在for循环中试图修改列表的值,可能会引发错误,其原因就在于Python的for循环存在自动填充,先看下面这个问题。
假设有一个列表,其中包含新注册但还未验证的用户;验证这些用户后,如何将他们移到另一个已验证用户列表呢?
用while循环处理列表
unconfirmed_users = ['alice', 'bob', 'cindy', 'deny']
confirmed_users = []
while unconfirmed_users:
current_user = unconfirmed_users.pop(0) # 每次将第一个元素弹出
confirmed_users.append(current_user)
print('Verifying user: ' + current_user.title())
#输出两个列表处理后的存储情况
print('\nThe following users have been confirmed:')
for confirmed_user in confirmed_users:
print(confirmed_user.title())
print('\nThe following users have not been confirmed:')
for unconfirmed_user in unconfirmed_users:
print(unconfirmed_user.title())
运行结果如下:
试着用for循环处理列表
unconfirmed_users = ['alice', 'bob', 'cindy', 'deny']
confirmed_users = []
for current_user in unconfirmed_users: # 语句1
confirmed_users.append(current_user)
unconfirmed_users.remove(current_user) # 语句2
print('Verifying user: ' + current_user.title())
print('\nThe following users have been confirmed:')
#输出两个列表处理后的存储情况
for confirmed_user in confirmed_users:
print(confirmed_user.title())
print('\nThe following users have not been confirmed:')
for unconfirmed_user in unconfirmed_users:
print(unconfirmed_user.title())
可以看到虽然我们遍历了列表并试图通过变量current_user(记为C_user)将unconfirmed_users(记为A_users)中的内容放入confirmed_users(记为B_users),但是失败了(只放入了部分元素),这是为什么?
其实,Python会根据实时的元素位置信息来选择变量下标。
开始在A_users中
[‘alice’, ‘bob’, ‘cindy’, ‘deny’] 对应的下标分别为0,1,2,3,变量C_user的下标的值为0,经过“语句2”的执行后把下标为0的元素(‘alice’)从A_users送到B_users,后来回到“语句1”,变量C_user的下标值变为1;
接下来在A_users中,[‘bob’, ‘cindy’, ‘deny’]对应的下标分别为0,1,2(下标信息实时刷新了),所以这时如果从A_users取出下标值为1的元素并送入B_users,便会把’cindy’送入,变量C_user的下标值变成2;
最后的的A_users包含[ ‘cindy’, ‘deny’]两个元素,对应的下标分别为0,1。由于变量C_user的下标值不在for循环的0,1范围内(下标值为2),所以直接退出循环,因此最终只从B_users中读取了2个元素。
试想一下有无解决方案呢?
仍然用for循环处理列表
unconfirmed_users = ['alice', 'bob', 'cindy', 'deny']
confirmed_users = []
for current_user in unconfirmed_users.copy(): #用方法copy()创建一个副本
confirmed_users.append(current_user)
unconfirmed_users.remove(current_user)
print('Verifying user: ' + current_user.title())
#输出两个列表处理后的存储情况
print('\nThe following users have been confirmed:')
for confirmed_user in confirmed_users:
print(confirmed_user.title())
print('\nThe following users have not been confirmed:')
for unconfirmed_user in unconfirmed_users:
print(unconfirmed_user.title())
这里我们通过 方法copy() 创建了一个A_users的副本,并让变量C_user在副本中读取元素信息,这样就不影响变量C_user在A_users中的下标值,这有点类似指针在函数中的使用。
下面是在for循环中使用 方法copy() 的原理简介:
A_users = unconfirmed_users(未验证用户)
B_users = confirmed_users(已验证用户)
C_user = current_user(for循环的变量)