中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Python列表嵌套常見坑點及解決方案

發布時間:2020-10-24 01:11:33 來源:腳本之家 閱讀:197 作者:lincappu 欄目:開發技術

1.嵌套列表

Python中有一種內置的數據類型叫列表(list),它是一種容器,可以用來承載其他的對象(準確的說是其他對象的引用),列表中的對象可以稱為列表的元素,很明顯我們可以把列表作為列表中的元素,這就是所謂的嵌套列表。

嵌套列表可以模擬出現實中的表格、矩陣、2D游戲的地圖(如植物大戰僵尸的花園)、棋盤(如國際象棋、黑白棋)等。

2.識別坑點

在使用嵌套的列表時要小心,否則很可能遭遇非常尷尬的情況,下面是一個小例子。

def main():  names = ['關羽', '張飛', '趙云', '馬超', '黃忠']  subjs = ['語文', '數學', '英語']  scores = [[0] * 3] * 5  for row, name in enumerate(names):    print('請輸入%s的成績' % name)    for col, subj in enumerate(subjs):      scores[row][col] = float(input(subj + ': '))  print(scores)if __name__ == '__main__':  main()
  names = ['關羽', '張飛', '趙云', '馬超', '黃忠']
  subjs = ['語文', '數學', '英語']
  scores = [[0] * 3] * 5
  for row, name in enumerate(names):
    print('請輸入%s的成績' % name)
    for col, subj in enumerate(subjs):
      scores[row][col] = float(input(subj + ': '))
  print(scores)
if __name__ == '__main__':
  main()

我們希望錄入5個學生3門課程的成績,于是定義了一個有5個元素的列表,而列表中的每個元素又是一個由3個元素構成的列表,這樣一個列表的列表剛好跟一個表格是一致的,相當于有5行3列。

接下來我們通過嵌套的for-in循環輸入每個學生3門課程的成績。程序執行完成后我們發現,每個學生3門課程的成績是一模一樣的(尷尬),而且就是最后錄入的那個學生的成績。

3。區分兩個概念

要想把這個坑填平,我們首先要區分對象和對象的引用這兩個概念,而要區分這兩個概念,還得先說說內存中的棧和堆。

我們經常會聽人說起“堆棧”這個詞,但實際上“堆”和“棧”是兩個不同的概念。眾所周知,一個程序運行時需要占用一些內存空間來存儲數據和代碼,那么這些內存從邏輯上又可以做進一步的劃分。

對底層語言(如C語言)有所了解的程序員大都知道,程序中可以使用的內存從邏輯上可以為五個部分,按照地址從高到低依次是:棧(stack)、堆(heap)、數據段(data segment)、只讀數據段(static area)和代碼段(code segment)。

棧用來存儲局部、臨時變量,以及函數調用時保存現場和恢復現場需要用到的數據,這部分內存在代碼塊開始執行時自動分配,代碼塊執行結束時自動釋放,通常由編譯器自動管理。

堆的大小不固定,可以動態的分配和回收,因此如果程序中有大量的數據需要處理,這些數據通常都放在堆上,如果堆空間沒有正確的被釋放會引發內存泄露的問題,而像Python、Java等編程語言都使用了垃圾回收機制來實現自動化的內存管理(自動回收不再使用的堆空間)。

4。小例子

所以,下面的代碼中,變量a并不是真正的對象,它是對象的引用,相當于記錄了對象在堆空間的地址,通過這個地址我們可以訪問到對應的對象。

a = object()b = ['apple', 'pitaya', 'grape']
b = ['apple', 'pitaya', 'grape']

同理,變量b是列表容器的引用,它引用了堆空間上的列表容器,而列表容器中并沒有保存真正的對象,它保存的也僅僅是對象的引用。

知道了這一點,我們可以回過頭看看剛才的程序,我們對列表進行[[0]* 3] * 5操作時,僅僅是將[0, 0, 0] 這個列表的地址進行了復制,并沒有創建新的列表對象。

所以,容器中雖然有5個元素,但是這5個元素引用了同一個列表對象。這一點可以通過id函數檢查scores[0]和scores[1]的地址得到證實。在此我們舉一個小例子,讀者朋友們可以敲一敲加深印象。

a = [[0]*3]*5id(a[0])id(a[1])# id相等
id(a[1])
# id相等

5。正確代碼

所以,正確的代碼應該按照如下的方式進行修改。

def main():  names = ['關羽', '張飛', '趙云', '馬超', '黃忠']  subjs = ['語文', '數學', '英語']  scores = [[]] * 5  for row, name in enumerate(names):    print('請輸入%s的成績' % name)    scores[row] = [0] * 3 #變為不再嵌套    for col, subj in enumerate(subjs):      scores[row][col] = float(input(subj + ': '))  print(scores)if __name__ == '__main__':  main()'關羽', '張飛', '趙云', '馬超', '黃忠']
  subjs = ['語文', '數學', '英語']
  scores = [[]] * 5
  for row, name in enumerate(names):
    print('請輸入%s的成績' % name)
    scores[row] = [0] * 3 #變為不再嵌套
    for col, subj in enumerate(subjs):
      scores[row][col] = float(input(subj + ': '))
  print(scores)
if __name__ == '__main__':
  main()

或者

def main():  names = ['關羽', '張飛', '趙云', '馬超', '黃忠']  subjs = ['語文', '數學', '英語']  scores = [[0] * 3 for _ in range(5)]  for row, name in enumerate(names):    print('請輸入%s的成績' % name)    scores[row] = [0] * 3    for col, subj in enumerate(subjs):      scores[row][col] = float(input(subj + ': '))  print(scores)if __name__ == '__main__':  main()
  names = ['關羽', '張飛', '趙云', '馬超', '黃忠']
  subjs = ['語文', '數學', '英語']
  scores = [[0] * 3 for _ in range(5)]
  for row, name in enumerate(names):
    print('請輸入%s的成績' % name)
    scores[row] = [0] * 3
    for col, subj in enumerate(subjs):
      scores[row][col] = float(input(subj + ': '))
  print(scores)
if __name__ == '__main__':
  main()

以上就是使用嵌套列表需要注意的問題及解決措施,希望大家多多總結,以此避免在使用嵌套列表或者復制對象時可能遇到的坑。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持億速云。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

澄迈县| 钟山县| 二连浩特市| 广德县| 子洲县| 舟曲县| 布尔津县| 库车县| 鄄城县| 五常市| 洛扎县| 仙桃市| 丹棱县| 囊谦县| 青浦区| 比如县| 高雄市| 海宁市| 松滋市| 揭阳市| 正安县| 龙州县| 沐川县| 南溪县| 石门县| 松潘县| 华容县| 达州市| 长岛县| 乌拉特后旗| 乌拉特中旗| 厦门市| 页游| 拉孜县| 屏南县| 宜春市| 清徐县| 武宣县| 海城市| 丘北县| 黑河市|