python赋值机制

2017-12-17  本文已影响0人  脏脏包盛
x = [1, 2, 3]
y = x
x[1] = 100
print(y)

[1, 100, 3]
改变变量x的值,变量y的值也随着改变,这与Python内部的赋值机制有关。

简单机制

先来看这一段代码在Python中的执行过程。

x = 500
y = x
y = 'foo'
内存 命名空间
pos1 : PyInt(500) (不可变) x : pos1
内存 命名空间
pos1 : PyInt(500) (不可变) x : pos1 y : pos1
内存 命名空间
pos1 : PyInt(500) (不可变) x : pos1
pos2 : PyStr('foo') (不可变) y : pos2

对这一过程进行验证,可以使用 id 函数。

id(x)

返回变量 x 的内存地址。

x = 500
id(x)

48220272L

y = x
id(y)

48220272L
也可以使用 is 来判断是不是指向同一个事物:

x is y

True
现在 y 指向另一块内存:

y = 'foo'
id(y)

39148320L

x is y

False

Python会为每个出现的对象进行赋值,哪怕它们的值是一样的,例如:

x = 500
id(x)

48220296L

y = 500
id(y)

48220224L

x is y

False
不过,为了提高内存利用效率,对于一些简单的对象,如一些数值较小的int对象,Python采用了重用对象内存的办法:

x = 2
id(x)

6579504L

y = 2
id(y)

6579504L

x is y

True

容器类型

现在来看另一段代码:

x = [500, 501, 502]
y = x
y[1] = 600
y = [700, 800]
内存 命名空间
pos1 : PyInt(500) (不可变)
pos2 : PyInt(501) (不可变)
pos3 : PyInt(502) (不可变)
pos4 : PyList(pos1, pos2, pos3) (可变) x : pos4
内存 命名空间
pos1 : PyInt(500) (不可变)
pos2 : PyInt(501) (不可变)
pos3 : PyInt(502) (不可变) x : pos4
pos4 : PyList(pos1, pos2, pos3) (可变) y: pos4
内存 命名空间
pos1 : PyInt(500)(不可变)
pos2 : 垃圾回收
pos3 : PyInt(502) (不可变)
pos4 : PyList(pos1, pos5, pos3) (可变) x : pos4
pos5 : PyInt(600) (不可变) y: pos4
内存 命名空间
pos1 : PyInt(500) (不可变)
pos3 : PyInt(502) (不可变)
pos4 : PyList(pos1, pos5, pos3) (可变)
pos5 : PyInt(600) (不可变)
pos6 : PyInt(700) (不可变)
pos7 : PyInt(800) (不可变) x : pos4
pos8 : PyList(pos6, pos7) (可变) y : pos8

对这一过程进行验证:

x = [500, 501, 502]
print( id(x[0]))
print (id(x[1]))
print (id(x[2]))
print (id(x))

48220224
48220248
48220200
54993032

赋值,id(y) 与 id(x) 相同。

y = x
print(id(y))

54993032

x is y

True

修改 y[1] ,id(y) 并不改变。

y[1] = 600
print (id(y))

54993032

id(x[1]) 和 id(y[1]) 的值改变了。

print(id(x[1]))
print(id(y[1]))

48220272
48220272

更改 y 的值,id(y) 的值改变

y = [700, 800]
print(id(y))
print(id(x))

54995272
54993032

上一篇 下一篇

猜你喜欢

热点阅读