Python list作为全局属性问题

Posted by Fioncat on September 3, 2021

以下两种写法:

1
2
3
4
5
6
class Numbers:
    nums = []
class Numbers:
    nums = None
    def __init__(self):
        self.nums = []

看上去是一样的,实际上是不一样的。

注意,python中的list是一种对象,它是一种实体。具名变量nums仅仅是指向它的一个引用。

在写法1中,nums指向的list仅在全局被初始化了一次。随后在创建Numbers对象时,会为每个对象创建一个新的nums引用指向这个全局list

在写法2中,每次创建一个新的对象,都会创建一个新的list实体,并使用nums指向它。

总结一下:

  • 写法1中,每个Numbers对象的nums引用指向同一个list。使用某个对象的nums来改变列表,会影响到其它对象。说得不准确点,就是所有Numbers对象共享一个nums列表。
  • 写法2中,每个Numbers对象持有自己的nums引用和自己的list,这些list实体之间是没有关系的,互不影响的。

执行以下代码:

1
2
3
4
5
6
7
8
9
10
nums1 = Numbers()
nums1.nums.append(1)
nums1.nums.append(2)

nums2 = Numbers()
nums2.nums.append(3)
nums2.nums.append(4)

print(f'{nums1.nums}')
print(f'{nums2.nums}')

写法1将会输出:

1
2
[1, 2, 3, 4]
[1, 2, 3, 4]

写法2输出:

1
2
[1, 2]
[3, 4]

一定一定要记住,list是对象,所有对list的赋值都是用一个新的引用去引用它,并不会增加新的list实体。只有调用listcopy()方法,才会创建一个新的list实体。