python把列表(list)傳給函數形參時的問題剖析


 

python把列表(list)傳給函數形參時的問題剖析

 國內圖書很大一部分關於“列表傳遞給函數”的問題避重就輕,有的甚至不會提及"列表傳參"的問題,業界強推的《流暢的python》一書有提及,且有專門的分析,另外,梁勇《python程序設計》也有類似分析,但是筆者感覺解釋的還是不夠通透,有必要重新剖析,

如果您是高手,請曬出您的評論留言,給出您的見解和剖析, 不勝感激.


 

預期結果

1 ---1---
2 [0, 1, 4, 9]
3 ---2---
4 [0, 1, 4, 9, 16]

實際執行結果

1 ---1---
2 [0, 1, 4, 9]
3 ---2---
4 [0, 1, 4, 9, 0, 1, 4, 9, 16]

 

出坑


 

當定義函數時,會保存函數中默認參數 list 的值,也就是列表 li=[];

在每次調用的時候如果傳遞了新的列表,則使用傳遞的列表,沒有傳遞,使用定義函數時保存的默認參數(li=[]);

上面兩次調用中,都沒有傳遞新的列表(使用默認列表 li=[] ),程序會調用定義函數時保存的默認參數((li=[]));

列表在append的時候會在 li=[] 原來的基礎上append追加值,所以會產生以上結果.

通過打印列表的ID進行辨識

打印列表 li=[] 的ID:

 1 def f(x,li=[]):
 2     print(id(li))  # 添加打印id
 3     for i in range(x):
 4         li.append(i*i)
 5     print(li)
 6  
 7  
 8 print('---1---')
 9 f(4)
10 print('---2---')
11 f(5)

 

結果:

1 ---1---
2 140306123906248
3 [0, 1, 4, 9]
4 ---2---
5 140306123906248
6 [0, 1, 4, 9, 0, 1, 4, 9, 16]

 

會發現ID值是相同的;  說明兩次執行時使用的都是定義函數時的默認參數 li=[ ]

執行時往里面傳新的列表

打印列表 li=[] 的ID 和 傳的新列表的ID:

 1 def f(x,li=[]):
 2     print(id(li))
 3     for i in range(x):
 4         li.append(i*i)
 5     print(li)
 6  
 7  
 8 print('---1---')
 9 f(4)
10 print('---2---')
11 f(5,[])
12 print('---3---')
13 f(6)

 

結果:

1 ---1---
2 [0, 1, 4, 9]
3 ---2---
4 [0, 1, 4, 9, 16]
5 ---3---
6 [0, 1, 4, 9, 0, 1, 4, 9, 16, 25]

 

會發現執行傳遞空(新)列表的函數時打印的ID不一樣,而沒有傳遞的一樣;

當傳遞空列表時,函數體當中會使用傳遞的空列表,沒有傳遞時,使用函數默認值 li=[ ], 所以會產生以上結果。

優化


 

如果想要達到預期的結果,只需要在函數體里進行判斷即可:

 1 def f(x, li=[]):
 2     if not li:
 3         # 如果li不為空的話,就往下走(清空列表); 為空就不走
 4         li = []
 5     for i in range(x):
 6         li.append(i * i)
 7     print(li)
 8 
 9 
10 print('---1---')
11 f(4)
12 print('---2---')
13 f(5)
14 print('---3---')
15 f(6)

 

結果:

1 ---1---
2 [0, 1, 4, 9]
3 ---2---
4 [0, 1, 4, 9, 16]
5 ---3---
6 [0, 1, 4, 9, 16, 25]

 ----over----

 

 

福利贈送: 代碼運行自動可視化網站,強烈推薦,不懂就把代碼扔進去,它會一步步教給您代碼運行的過程。

網址: http://pythontutor.com/

 

 

 

 

 

參考修改:https://www.cnblogs.com/gpd-Amos/p/8998059.html


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM