Python 板


LINE

: 推 pmove: 回b大,你說的官方是出自: 04/02 09:29 : → pmove: https://docs.python.org/3.3/tutorial/controlflow.html 04/02 09:29 : → pmove: Important warning: The default value is evaluated only 04/02 09:30 : → pmove: 下一句就是:This makes a difference when the default is 04/02 09:30 : → pmove: a mutable object such as a list, dictionary, or 04/02 09:31 : → pmove: instances of most classes. 04/02 09:31 : → pmove: 裡面就提到mutable object.你要覺得倒果為因,我也沒辦法 04/02 09:33 : 推 pmove: 我自己是覺得你跟我講The default value is evaluated only 04/02 09:53 : → pmove: once" 我自己是沒辦法理解的,但是你告訴我哪些是mutable? 04/02 09:54 : → pmove: 哪些是immutable? mutable/immutable會有哪種情形?這樣我 04/02 09:54 : → pmove: 比較好理解。所以才從mutable/immutable切入。 04/02 09:55 : 推 froce: 推樓上,這根本就不是bug,動態語言常這樣設計 04/06 07:44 看推文似乎很多人分不清楚值(value)和表達式(expression) 舉個例子 1 def connect_db(): 2 return ... 3 4 def work(db = connect_db()): 5 db.execute(...) 6 7 DB = connect_db() 8 work( DB ) 請問 connect_db() 會執行幾次? 正常人類的思考: 嗯... work 函數需要 1 個叫做 db 參數,我也提供一個正確的值, 我不在乎有沒有預設值, 反正我沒用到, 那麼 connect_db() 應該只在第 7 行處執行了一次 但 Python 實際上會執行兩次 第一次在第 4 行 第二次在第 7 行 這就是官方文檔中 "The default values are evaluated at the point of function definition" 的意思 看到了嗎? connect_db 的回傳值是 mutable 或 immutable 跟本不重要 究其原因就只是 Python 的規格要求:參數的預設值於函數定義時計算 CPython 也只是照著規格書去實作而已 再舉個例子, 這次預設值連型別都沒有: 1 def check_missing(): 2 raise Exception("You missed one argument") 3 4 def work(db = check_missing()): 5 db.execute(...) 6 7 DB = connect_db() 8 work( DB ) 請問以上程式可否順利執行? 正常人類: 當然可以 (事實上這就是很多動態語言檢查「參數是否缺少」的作法) Python: 不行 (因為 check_missing() 在第 4 行就呼叫了) Traceback (most recent call last): File "test.py", line 4, in <module> def work(db = check_missing()): File "test.py", line 2, in check_missing raise Exception("You missed one argument") Exception: You missed one argument 你看, 我連函數都還沒呼叫就掛了! 為什麼說這是個雷? 因為這個特性簡直莫名奇妙, 完全反直覺, 也沒有什麼好處. 有人說動態語言常這樣設計, 純屬胡說八道 Javascript: function check_missing(){ throw "Error" } function work(db=check_missing()){ ... } work(db) 順利執行 C++: int f(){ throw "error"; } int work(int data=f()){ return data; } int main(){ work(3); } 順利執行 Ruby: def check_missing(number) raise 'An error has occured' end def work(data = check_missing() ) puts(data) end work(1) 順利執行 PHP: 預設值只能使用字面常量(literals) Lua: 不支援預設值 Java: 不支援預設值 就我知道的語言中只有 Python 有這種設計, 它沒有帶來任何好處. 這還只是第一個雷而已. 讓我們看第二個雷 "The default value is evaluated only once." 就是這個預設值不但會「超前計算」, 而且還會被 cache 住. 原po的問題就是這個特性造成的. 推文裡一直說因為原po的預設值[]是mutable, 因此如何如何, 這裡給一個 immutable 一樣會雷到人的例子 1 import random 2 3 def func(a=random.random()): 4 return a 5 6 x = func() 7 y = func() 請問 x 和 y 會相等嗎? 正常人類: 機率很小, 除非剛好兩次隨機抽到一樣的數 Python : 保證 x,y 完全一樣! 為什麼? 因為 Python 會把 random.random() 的值記錄在 func.__defaults__ 裡 所以你看到了, random.random() 的回傳值 float 是 immutable 又如何? 還不是雷人. 雷就雷吧, 有什麼其他的好處嗎? 抱歉, 只有特定場合需要建立變數的副本時很方便, 但這屬於「歪打正著」和「先射箭再畫靶」的劣招, 帶來方便的同時帶來更多混淆. 這兩個雷就是設計缺陷, 沒什麼好爭的, 從 python 2 或更早以前就存在, Guido 本人也承認是設計缺陷 https://twitter.com/gvanrossum/status/1014524798850875393 正確的實作應該是「預設值僅在 函數呼叫且該參數被省略時 計算」 這是 Python 少數幾個地雷之一, 小心並正確地避開就是了 官方建議「預設值不要用mutable object」也是如此用意. 但這是不夠的, 你還要保證這個 default value 建立時沒有其他「副作用」 另外 其實不建議初學者直接往所謂的「底層」去看, 那不是原因, 那只是實作而已. 就像寫C++也不要沒事就看組語, 寫Java不要沒事就去讀bytecode 應先以更宏觀角度理解現象的本質 你以為把「基礎」研究透了, 實際上你只是研究了一個二十年前的錯誤而已, 甚至無法意識到這究竟是不是個錯誤. --



※ 發信站: 批踢踢實業坊(ptt.cc), 來自: 172.103.227.117 (加拿大)
※ 文章網址: https://webptt.com/m.aspx?n=bbs/Python/M.1586154248.A.888.html ※ 編輯: bibo9901 (172.103.227.117 加拿大), 04/06/2020 14:25:17 ※ 編輯: bibo9901 (172.103.227.117 加拿大), 04/06/2020 14:59:30 ※ 編輯: bibo9901 (172.103.227.117 加拿大), 04/06/2020 14:59:59 ※ 編輯: bibo9901 (172.103.227.117 加拿大), 04/06/2020 15:01:07 ※ 編輯: bibo9901 (172.103.227.117 加拿大), 04/06/2020 15:02:11 ※ 編輯: bibo9901 (172.103.227.117 加拿大), 04/06/2020 15:13:05
1F:→ pmove: 第8行work(DB)可以拿掉,仍然會執行第4行根第7行兩次 04/06 15:07
※ 編輯: bibo9901 (172.103.227.117 加拿大), 04/06/2020 15:14:05
2F:→ bibo9901: 我當然知道可以拿掉 我是以人類的角度寫正常的程式 04/06 15:14
※ 編輯: bibo9901 (172.103.227.117 加拿大), 04/06/2020 15:17:53 ※ 編輯: bibo9901 (172.103.227.117 加拿大), 04/06/2020 15:19:28
3F:→ pmove: 你舉例的情形是argument assign一個function, 04/06 15:39
※ 編輯: bibo9901 (172.103.227.117 加拿大), 04/06/2020 15:39:52
4F:→ pmove: function雖然會return值,但跟argument直接assign mutable/ 04/06 15:40
※ 編輯: bibo9901 (172.103.227.117 加拿大), 04/06/2020 15:40:43
5F:→ pmove: immutable時的情況,是不一樣的。原Po是問assign None obje 04/06 15:41
6F:→ pmove: t, None object is a unique, immutable object. 04/06 15:41
※ 編輯: bibo9901 (172.103.227.117 加拿大), 04/06/2020 15:45:25
7F:→ bibo9901: 我哪有給一個function, 我是給function的回傳值 OK? 04/06 15:47
8F:→ bibo9901: 原po的問題還結合了 variable scope 的問題 04/06 15:48
※ 編輯: bibo9901 (172.103.227.117 加拿大), 04/06/2020 15:50:36
9F:→ bibo9901: 難道random.random()就不是unique&immutable object XD 04/06 16:03
10F:→ bibo9901: 再說unique object是啥? 每個object都嘛unique.. 04/06 16:04
11F:→ pmove: Unique 是指None object的值只有None, 不像int可以有1 或者 04/06 16:31
12F:→ pmove: 4... 04/06 16:31
13F:→ pmove: 我覺得b大有點超譯原po所要問的,只有我這樣感覺嗎? 04/06 16:46
14F:推 yushes920179: 我覺得你說的很好懂 不過有必要這麼派嗎 04/07 08:33
15F:推 TuCH: 學到了 重點就是"參數的預設值於函數定義時計算" 04/07 15:57
16F:→ TuCH: 雷一跟雷二跟其他例子都是這個的延伸 04/07 15:59
17F:→ pmove: 補充一點,如果函數定義在縮排裡面,但是此函數的參數直接 04/07 16:36
18F:→ pmove: 函數。那麼此參數就不會立刻有值了。 04/07 16:37
19F:推 Starcraft2: 推詳細說明 04/09 01:54
20F:推 s860134: "參數的預設值於函數定義時計算" 重點 04/15 22:55
21F:→ s860134: 只要你預設值不要存值,存 function pointer 就解決了~ 04/15 22:57
22F:→ s860134: e.g. property, 或是 callable object 04/15 22:58
23F:→ s860134: 像本篇例子 check_missing 可以置換成一個有 __call__ 04/15 23:00
24F:→ s860134: 或 __getattr__ 的自定義 class 即可達成想要的效果 04/15 23:02







like.gif 您可能會有興趣的文章
icon.png[問題/行為] 貓晚上進房間會不會有憋尿問題
icon.pngRe: [閒聊] 選了錯誤的女孩成為魔法少女 XDDDDDDDDDD
icon.png[正妹] 瑞典 一張
icon.png[心得] EMS高領長版毛衣.墨小樓MC1002
icon.png[分享] 丹龍隔熱紙GE55+33+22
icon.png[問題] 清洗洗衣機
icon.png[尋物] 窗台下的空間
icon.png[閒聊] 双極の女神1 木魔爵
icon.png[售車] 新竹 1997 march 1297cc 白色 四門
icon.png[討論] 能從照片感受到攝影者心情嗎
icon.png[狂賀] 賀賀賀賀 賀!島村卯月!總選舉NO.1
icon.png[難過] 羨慕白皮膚的女生
icon.png閱讀文章
icon.png[黑特]
icon.png[問題] SBK S1安裝於安全帽位置
icon.png[分享] 舊woo100絕版開箱!!
icon.pngRe: [無言] 關於小包衛生紙
icon.png[開箱] E5-2683V3 RX480Strix 快睿C1 簡單測試
icon.png[心得] 蒼の海賊龍 地獄 執行者16PT
icon.png[售車] 1999年Virage iO 1.8EXi
icon.png[心得] 挑戰33 LV10 獅子座pt solo
icon.png[閒聊] 手把手教你不被桶之新手主購教學
icon.png[分享] Civic Type R 量產版官方照無預警流出
icon.png[售車] Golf 4 2.0 銀色 自排
icon.png[出售] Graco提籃汽座(有底座)2000元誠可議
icon.png[問題] 請問補牙材質掉了還能再補嗎?(台中半年內
icon.png[問題] 44th 單曲 生寫竟然都給重複的啊啊!
icon.png[心得] 華南紅卡/icash 核卡
icon.png[問題] 拔牙矯正這樣正常嗎
icon.png[贈送] 老莫高業 初業 102年版
icon.png[情報] 三大行動支付 本季掀戰火
icon.png[寶寶] 博客來Amos水蠟筆5/1特價五折
icon.pngRe: [心得] 新鮮人一些面試分享
icon.png[心得] 蒼の海賊龍 地獄 麒麟25PT
icon.pngRe: [閒聊] (君の名は。雷慎入) 君名二創漫畫翻譯
icon.pngRe: [閒聊] OGN中場影片:失蹤人口局 (英文字幕)
icon.png[問題] 台灣大哥大4G訊號差
icon.png[出售] [全國]全新千尋侘草LED燈, 水草

請輸入看板名稱,例如:Tech_Job站內搜尋

TOP