慎用 Python list 乘法

2020-07-14 14:53:19 浏览数 (1)

误用 list 乘法

今天刷 LeetCode 碰到一个水题转置矩阵, 这不就是先生成个空的倒置矩阵再填结果嘛,没多想就用 list 乘法上手就写。

代码语言:javascript复制
class Solution:
    def transpose(self, A):
        """
        :type A: List[List[int]]
        :rtype: List[List[int]]
        """
        x, y = len(A), len(A[0])

        buff =[[0] * x] * y
        for i in range(x):
            for j in range(y):
                buff[j][i] = A[i][j]
        return buff

看似结果非常正确,但是样例输出了一个很奇怪的结果。

代码语言:javascript复制
我的输入:
    [[1,2,3],[4,5,6]]

我的答案:
    [[3,6],[3,6],[3,6]]

标准答案:
    [[1,4],[2,5],[3,6]]

赶紧在第12行前加上print(buff)一看

代码语言:javascript复制
我的输入:
    [[1,2,3],[4,5,6]]

标准输出:
    [[0, 0], [0, 0], [0, 0]]
    [[1, 0], [1, 0], [1, 0]]
    [[2, 0], [2, 0], [2, 0]]
    [[3, 0], [3, 0], [3, 0]]
    [[3, 4], [3, 4], [3, 4]]
    [[3, 5], [3, 5], [3, 5]]

我的猜测与验证

发现果然列表里的每一个子元素都相等了,猜测可能是 只复制了值的引用,而不是新建了一个对象,接下来就是验证。

代码语言:javascript复制
>>> a = [[0]*6]*4
>>> a
[[0, 0, 0, 0, 0, 0], [0, 0, 0, 0, 0, 0], [0, 0, 0, 0, 0, 0], [0, 0, 0, 0, 0, 0]]
>>> id(a[0])
1423765355080
>>> id(a[1])
1423765355080

发现不出所料果然是这样。

正确姿势

list 的乘法是能很方便的构建一个全为重复元素的一维列表方法,但在多维情况下非常容易出错。 所以说慎用 list 乘法!!!想构建 list 老老实实给我用列表生成器去。不要嫌写的太多不 pythonic

代码语言:javascript复制
class Solution:
    def transpose(self, A):
        """
        :type A: List[List[int]]
        :rtype: List[List[int]]
        """
        x, y = len(A), len(A[0])

        buff =[[0 for i in range(x)] for i in range(y)]
        for i in range(x):
            for j in range(y):
                buff[j][i] = A[i][j]
        return buff

0 人点赞