大概兩年半前,我萌生了要創(chuàng)作一個新的系列文章的想法徒爹,也就是“Python為什么”荚醒,試圖對 Python 的語法及特性提出“為什么”式的問題,以此加深對它的理解隆嗅,探尋使用技巧界阁、發(fā)展演變、設計哲學等話題胖喳。
一直以來泡躯,我都是一個有著較強問題意識的充滿著好奇心的人,擅長于識別出相似東西的差異,并從差異性上發(fā)現(xiàn)事物的獨特意義较剃。
于是咕别,當將 Python 與其它編程語言作比較時,加上閱讀及翻譯了一些 PEP 從而積攢了一些素材后写穴,我就得到了很多的小發(fā)現(xiàn)惰拱。當確認了國內(nèi)外的技術社區(qū)里缺少這方面的文章后,我就更確信了這件事的獨特價值啊送。
我當時有個天真的想法偿短,覺得可以按照“十萬個為什么”的方式,寫出源源不斷的文章……
剛開始的 2020 年下半年馋没,我創(chuàng)作力旺盛昔逗,寫了約 20 篇“Python為什么”系列文章!然而篷朵,到了 2021 年勾怒,僅有 2 篇!再到 2022 年款票,也是僅僅 2 篇!泽论!……
時間都去哪兒了艾少?怎么我才稍稍微偷了個懶兒,它就不見了呢翼悴?本來計劃有不少想寫的話題的缚够,怎么拖著拖著就忘了該怎么寫了呢……
最近眼看到了年末,我越想越是有些不甘鹦赎,于是谍椅,花了幾天時間,好好梳理了下“Python為什么”系列文章古话,優(yōu)化了 Github 的介紹內(nèi)容雏吭,準備認真把這個系列重拾起來!
我把之前調(diào)查問卷里遺留的問題陪踩,以及其它計劃要寫的話題放在 Issues 跟蹤杖们,歡迎大家來提問題/給建議/指導寫作/監(jiān)督催更……
下面放出的是目前系列文章的介紹,懇請喜歡本系列的同學給顆 star 鼓勵一下<缈瘛(內(nèi)容會不斷更新/增長摘完,請以 Github 主頁為準。)
如果你在手機微信端閱讀傻谁,由于鏈接跳轉(zhuǎn)麻煩孝治,建議你通過這個合集的鏈接進行閱讀。
文章列表
-
Python 設計和歷史的常見問題
- Python 官方提供了約 30 個常見問題的 FAQ,你可以從中快速得到“權(quán)威”的解釋
-
Python 為什么用 len() 函數(shù)谈飒,不用 x.len() 風格岂座?
- 介紹了《流暢的Python》及 Guido 的解釋
- 我本人認為這體現(xiàn)了 Python 對世界本質(zhì)的洞察
- 文章順便回答了:為什么 Python 的索引從 0 開始計數(shù)?
-
Python 為什么使用縮進來劃分代碼塊步绸?
- 這是個經(jīng)典的問題掺逼,總會被提起瓤介,我總結(jié)了 8 個原因
- 有不少人對上述 8 個原因并不買賬吕喘,因此我補充了一個回復:Python 的縮進絕不是反人類的設計!
- Guido 在一次采訪中說:嚴格要求代碼縮進確實有點夸張刑桑,改用花括號氯质,也不是不可以
- Python 的縮進起源于 ABC,而 ABC 的縮進起源于 60-70 年代的編程暢想
-
Python 為什么不用分號作語句終止符祠斧?
- 分號一般有分隔符與終止符兩種作用假颇,但 Python 只用分號作為分隔符鸵赫,卻不用它作為終止符, 而是改用換行作為終止符。本文精煉總結(jié)了 5 個原因
-
Python 為什么沒有 main 函數(shù)拟糕?為什么我不推薦寫 main 函數(shù)?
- main 函數(shù)作為某些編程語言的執(zhí)行入口是強制必要的奴烙,然而 Python 這門腳本語言有著自己更為靈活的執(zhí)行方式
- 在我的編程習慣中剂娄,我反感那些不假思索的
if __name__ == '__main__'
寫法,文中給出了我的編程建議
-
Python 為什么推薦蛇形命名法鲸阻?
- 編程語言中有好幾種變量命名風格跋涣,最為流行的兩種分別是駝峰命名法和蛇形命名法。本文從編程語言的歷史發(fā)展過程和語言內(nèi)部的使用習慣角度鸟悴,解釋了為什么 Python 更偏好于蛇形命名法
-
Python 為什么不支持 i++ 自增語法陈辱,不提供 ++ 操作符?
- 有過 C/C++/Java 等語言的編程經(jīng)驗的開發(fā)者會疑惑细诸,為什么 Python 中沒有 i++ 這樣的語法
- 這個問題反映出 Python 中的數(shù)字對象跟其它語言中的數(shù)字有著根本性的差異沛贪;另外,Python 的可迭代對象特性震贵,也深刻影響著語言的諸多設計方面
-
Python 為什么只需一條語句“a,b=b,a”鹏浅,就能直接交換兩個變量?
- 很多人以為“a,b=b,a”(交換變量操作)跟“a,b=1,2”(多變量賦值)一樣屏歹,都是基于元組解包的特性隐砸,然而 CPython 的實現(xiàn)并非如此
- CPython 使用專門的優(yōu)化指令(即 ROT_TWO、ROT_THREE 和 ROT_FOUR)實現(xiàn)棧頂元素的快捷交換
- 當同時交換的元素數(shù)量大于 4 個時蝙眶,解釋器才會跟“a,b=1,2”(多變量賦值)一樣季希,基于解包實現(xiàn)變量賦值
-
Python 為什么用 # 號作注釋符褪那?
- 注釋符是編程語言中最基礎的要素之一,Python 屬于“# 號注釋符陣營”式塌,原因或許是它遵循著 Shell 等腳本語言的傳統(tǒng)
- Python 中不存在“塊注釋符”博敬,Guido 曾建議使用多行字符串(multi-line strings)來達到塊注釋的效果,但這種方案在語義上有點怪異
-
Python 為什么要有 pass 語句峰尝?
- pass 是 Python 獨有的一種空操作偏窝,其它語言并沒有這樣的設計
- pass 可以作為一種空間占位符,輔助程序員快速編程武学,然而這點小用途并非至關重要的
- 由于 Python 不使用花括號之類的手段來劃分代碼塊祭往,因此在定義空函數(shù)時,pass 就成了一種補齊語法邏輯的方案
-
Python 為什么會有個奇怪的“...”對象火窒?
- ... 是 Python3 在 PEP-3100 中引入的一個內(nèi)置常量硼补,與 Ellipsis 表示同一個對象
- 官方說它們是單例的,然而這有違事實熏矿。要么是文檔錯了已骇,要么這是一個 Bug ?
- ... 有什么用處票编,能夠解決什么問題褪储?文中介紹了 4 個用途:擴展切片語法、表達“未完成的代碼”語義慧域、Type Hint 用法鲤竹、表示無限循環(huán)
-
Python 為什么能支持任意的真值判斷?
- 這也是 Python 與眾不同的一個特性吊趾,它將其它語言中僅限于布爾類型的操作(if 或 while 或布爾操作 and宛裕、or瑟啃、not)论泛,擴展到了任意對象,帶來了極大的靈活性
- 真值判斷的結(jié)果取決于__bool__() 和 __len__() 這兩個魔術方法的返回值
- Python 甚至可以對數(shù)字對象作真值判斷(表示 0 的數(shù)為 False蛹屿,其它數(shù)為 True)
-
Python 函數(shù)為什么會默認返回 None屁奏?
- Python 隱性地為沒有帶 return 的函數(shù)添加一個 return 操作,即默認返回 None 值错负,這是由解釋器強行注入的邏輯坟瓢。這意味著:Python 中不存在無返回值的函數(shù)
- 為什么 Python 要強制令所有函數(shù)都有一個返回值呢?為什么它不支持無返回值的空函數(shù)呢犹撒?
-
Python 為什么沒有 void 關鍵字折联?
- void 通常指的是一種類型(type),但是它沒有具體的值(value)识颊。文中介紹了其它語言需要使用 void 關鍵字實現(xiàn)的兩種功能
- Python 舍棄了表示“沒有值的類型”的 void诚镰,統(tǒng)一使用表示“僅有一個值的類型” None奕坟,配合前一篇“所有函數(shù)必然有返回值”的設計,實現(xiàn)了簡單好用的效果
-
Python 為什么是強類型語言清笨,不是弱類型語言月杉?
- 動靜類型與強弱類型是兩組不同維度的概念,不應混為一談抠艾。在編程語言發(fā)展的早期苛萎,當強弱類型的概念還未提出時,一些大佬使用動靜類型來籠統(tǒng)地描述語言的特性检号,這是歷史原因
- 如今主流觀點以“隱式類型轉(zhuǎn)換”來劃分強弱類型腌歉,Python 毫無疑問是強類型語言。文中針對幾個易混淆的問題谨敛,詳細解釋了為什么 Python 中不存在“隱式類型轉(zhuǎn)換”
-
Python 之父為什么嫌棄 lambda 匿名函數(shù)究履?
- lambda 語法借鑒自 lisp 語言,卻遭到 Python 之父的嫌棄脸狸,然而它竟從他的屠刀下幸存最仑,這段故事充滿戲劇性
- Python 的 lambda 只支持單行表達式,功能不完備炊甲。曾有人提議增強 lambda 語法泥彤,Python 之父認為那不是好的設計,因而否決了
- Guido 提出要一次性移除 reduce()卿啡、map()吟吝、filter() 以及 lambda,但最后他妥協(xié)了
-
Python 為什么不支持 switch 語句颈娜?
- 大多數(shù)語言都提供了 switch 語句或者極其相似的東西剑逃,但在 Python 之父的裁決下,Python 不提供 switch 語句
- 文章介紹了試圖引入 switch 語句的 PEP-275 與 PEP-3103官辽,總結(jié)了這兩個提案的要點以及被否決的原因
-
Python 疑難問題:[] 與 list() 哪個快蛹磺?為什么快?快多少呢同仆?
- 兩種創(chuàng)建列表的 [] 與 list() 寫法萤捆,哪一個更快呢,為什么它會更快呢俗批?
- 文章通過字節(jié)碼與執(zhí)行過程的分析俗或,解釋了兩者執(zhí)行速度的差異
-
為什么說 Python 內(nèi)置函數(shù)并不是萬能的?
- 內(nèi)置函數(shù)的名稱并不是關鍵字岁忘,而內(nèi)置作用域位于名稱查找的最低優(yōu)先級辛慰,因此在調(diào)用時,某些內(nèi)置函數(shù)/類型的執(zhí)行速度就明顯慢于它們對應的字面量表示法
-
為什么繼承 Python 內(nèi)置類型會出問題干像?帅腌!
- 由《流暢的Python》中的例子辱志,引出 Python 在內(nèi)置類型子類化時不合常理的話題
- 分析魔術方法的底層實現(xiàn)邏輯及調(diào)用關系,解釋內(nèi)置類型存在的問題
- 介紹了內(nèi)置類型子類化的最佳實踐
-
為什么 Python 的 f-string 可以拼接字符串與數(shù)字狞膘?
- Python 是強類型語言揩懒,在不經(jīng)過強制類型轉(zhuǎn)換的情況下,字符串無法拼接數(shù)字
- 介紹了 PEP-498 實現(xiàn) f-string 的原理
-
Python 的切片為什么不會索引越界挽封?
- 切片是不少編程語言的特性已球,Python 的切片不僅功能完善,而且在使用上更為靈活
- 索引越界是一個常見的問題辅愿,Python 切片使用了幾條規(guī)則智亮,屏蔽了可能導致出錯的情況
- 文章介紹了 Python 的解決方案,但是也留下了一個疑問:為什么 Python 的切片語法要允許索引超出邊界呢点待,為什么不設計成拋出索引錯誤阔蛉?
-
為什么 range() 生成的不是迭代器?
- 有很多內(nèi)置方法可以生成迭代器癞埠,然而似乎只有 range() 生成的是可迭代對象状原,這個 range() 顯得非常獨特。文中給出了我對此的猜想
- 我還注意到 range 是一種不可變序列苗踪,然而它跟字符串這種不可變序列相比颠区,也有著獨特的表現(xiàn)
-
Python 為什么要保留顯式的 self ?
- 這也是一個常見問題通铲。這里給出了官方文檔的解釋毕莱,另外附了 Guido 的一篇博客全文
-
Python 為什么不設計 do-while 循環(huán)結(jié)構(gòu)?
- 在 C/C++颅夺、C#朋截、PHP、Java吧黄、JavaScript 等語言中部服,do-while 是一種基本結(jié)構(gòu)。Python 為什么不沿襲它們的傳統(tǒng)呢稚字?有什么特殊的考慮饲宿?
- 文章列舉了其它語言中 do-while 語法的主要使用場景厦酬,解釋了為什么 Python 可以不用這種結(jié)構(gòu)
- 介紹了 PEP-315 試圖引入 do-while 結(jié)構(gòu)的嘗試胆描,以及 Guido 的反對意見
-
為什么 Python 3 把 print 改為函數(shù)?
- Python3 與 Python2 最顯眼的一個區(qū)別就是:print 語句變成了 print() 函數(shù)
- PEP-3105 Make print a function 是對這個問題最好的回答
-
為什么說 Python 最會變魔術的魔術方法是它仗阅?
- __missing__() 是僅在內(nèi)置類型的子類上才存在的魔術方法昌讲,似乎是唯一的特例
- __missing__() 極為特殊,Python 解釋器為它開了后門减噪,實現(xiàn)了最為罕見的“魔術方法間調(diào)用”邏輯
-
Python 為什么用”elif“短绸,而不是“else if”车吹?
- elif 寫法相比于“else if”更為簡潔,這種寫法并非 Python 首創(chuàng)醋闭。Guido 發(fā)推特解釋了這種寫法的來源
當在兩年半前寫下第一篇“Python為什么”系列的時候窄驹,我無法想象自己會在 2023 年到來之際寫下這一篇宣告重新起航的小結(jié),更無法想象是在下一個兩年半证逻,或者五年半或者更久乐埠,再次寫下一篇新的總結(jié)。誰說得準呢囚企!
但是丈咐,不忘初心,珍惜當下的決心龙宏,樹立砥礪前行的恒心棵逊,我可以的!
最后银酗,別急著劃走啊辆影,請一定記得點個關注、點個 star 哈黍特,喵喵喵~~