在Python中,令values=[0,1,2];values[1]=values,为何成果是[0,

    添加时间:2013-7-20 点击量:
    转载自:http://www.zhihu.com/question/21000872/answer/16856382
    >>>
    values = [0 1 2]
    >>> values[1] = values
    >>> values
    [0 [...], 2]
    我预想该当是 
    [0 [0 1 2], 2]


    为何要赋值无穷次?


    Python 没有赋值,只有引用。你如许相当于创建了一个引用自身的布局,所以导致了无穷轮回。为了懂得这个题目,有个根蒂根基概念须要搞清楚。

    Python 没有「变量」,我们日常平凡所说的变量其实只是「标签」。履行 



    values = [0 1 2]
    


    的时辰,Python 做的工作是起建一个列表对象 [0, 1, 2],然后给它贴上名为 values 的标签。若是随后又履行



    values = [3 4 5]
    


    的话,Python 做的工作是创建另一个列表对象 [3, 4, 5],然后把刚才那张名为 values 的标签畴前面的 [0, 1, 2] 对象上撕下来,从头贴到 [3, 4, 5] 这个对象上。

    至始至终,并没有一个叫做 values 的列表对象容器存在,Python 也没有把任何对象的值复制进 values 去。过程如图所示:


    履行



    values[1] = values
    


    的时辰,Python 做的工作则是把 values 这个标签所引用的列表对象的第二个元素指向 values 所引用的列表对象本身。履行完毕后,values 标签还是指向本来那个对象,只不过那个对象的布局产生了变更,从之前的列表 [0, 1, 2] 变成了 [0, ?, 2],而这个 ? 则是指向那个对象本身的一个引用。如图所示:

    要达到你所须要的结果,即获得 [0, [0, 1, 2], 2] 这个对象,你不克不及直接将 values[1] 指向 values 引用的对象本身,而是须要吧 [0, 1, 2] 这个对象「复制」一遍,获得一个新对象,再将 values[1] 指向这个复制后的对象。Python 里面复制对象的操纵因对象类型而异,复制列表 values 的操纵是



    values[:]
    


    所以你须要履行



    values[1] = values[:]
    


    Python 做的工作是,先 dereference 获得 values 所指向的对象 [0, 1, 2],然后履行 [0, 1, 2][:] 复制操纵获得一个新的对象,内容也是 [0, 1, 2],然后将 values 所指向的列表对象的第二个元素指向这个复制二来的列表对象,终极 values 指向的对象是 [0, [0, 1, 2], 2]。过程如图所示:


    往更深处说,values[:] 复制操纵是所谓的「浅复制」(shallow copy),当列表对象有嵌套的时辰也会产生出乎料想的错误,比如



    a = [0 [1 2], 3]
    
    b = a[:]
    a[0] = 8
    a[1][1] = 9


    问:此时 a 和 b 分别是几许?

    正确答案是 a 为 [8, [1, 9], 3],b 为 [0, [1, 9], 3]。发明没?b 的第二个元素也被改变了。想想是为什么?不熟悉打听的话看下图


    正确的复制嵌套元素的办法是进行「深复制」(deep copy),办法是



    import copy
    

    a = [0 [1 2], 3]
    b = copy.deepcopya
    a[0] = 8
    a[1][1] = 9


    读书,不要想着实用,更不要有功利心。读书只为了自身的修养。邂逅一本好书如同邂逅一位知己,邂逅一个完美之人。有时心生敬意,有时怦然心动。仿佛你心底埋藏多年的话,作者替你说了出来,你们在时光深处倾心相遇的一瞬间,情投意合,心旷神怡。
    分享到: