Python 為什麼不支持 i++ 自增語法,不提供 ++ 操作符?

来源:https://www.cnblogs.com/pythonista/archive/2020/06/21/13174427.html
-Advertisement-
Play Games

在 C/C++/Java 等等語言中,整型變數的自增或自減操作是標配,它們又可分為首碼操作(i 和 --i)與尾碼操作(i 和 i--),彼此存在著一些細微差別,各有不同的用途。 這些語言的使用者在接觸 Python 時,可能會疑惑為什麼它不提供 ++ 或 -- 的操作呢?在我前不久發的《Pytho ...


在 C/C++/Java 等等語言中,整型變數的自增或自減操作是標配,它們又可分為首碼操作(++i 和 --i)與尾碼操作(i++ 和 i--),彼此存在著一些細微差別,各有不同的用途。

這些語言的使用者在接觸 Python 時,可能會疑惑為什麼它不提供 ++ 或 -- 的操作呢?在我前不久發的《Python的十萬個為什麼?》里,就有不少同學在調查問卷中表示了對此話題感興趣。

Python 中雖然可能出現 ++i 這種首碼形式的寫法,但是它並沒有“++”自增操作符,此處只是兩個“+”(正數符號)的疊加而已,至於尾碼形式的“++”,則完全不支持(SyntaxError: invalid syntax)。

本期“Python為什麼 ”欄目,我們將會從兩個主要的角度來回答:Python 為什麼不支持 i++ 自增語法? (PS:此處自增指代“自增和自減”,下同)

首先,Python 當然可以實現自增效果,即寫成i += 1 或者 i = i + 1 ,這在其它語言中也是通用的。

雖然 Python 在底層用了不同的魔術方法(__add__()__iadd__() )來完成計算,但錶面上的效果完全相同。

所以,我們的問題可以轉化成:為什麼上面的兩種寫法會勝過 i++,成為 Python 的最終選擇呢?

1、Python 的整數是不可變類型

當我們定義i = 1000 時,不同語言會作出不同的處理:

  • C 之類的語言(寫法 int i = 1000)會申請一塊記憶體空間,並給它“綁定”一個固定的名稱 i,同時寫入一個可變的值 1000。在這裡,i 的地址以及類型是固定的,而值是可變的(在一定的表示範圍內)
  • Python(寫法i = 1000)也會申請一塊記憶體空間,但是它會“綁定”給數字 1000,即這個 1000 的地址以及類型是固定的(immutable),至於 i,只是一個名稱標簽貼在 1000 上,自身沒有固定的地址和類型

所以當我們令 i “自增”時(i = i + 1),它們的處理是不同的:

  • C 之類的語言先找到 i 的地址上存的數值,然後令它加 1,操作後新的數值就取代了舊的數值
  • Python 的操作過程是把 i 指向的數字加 1,然後把結果綁定到新申請的一塊記憶體空間,再把名稱標簽 i “貼”到新的數字上。新舊數字可以同時存在,不是取代關係

打一個不太恰當的比方:C 中的 i 就像一個宿主,數字 1000 寄生在它上面;而 Python 中的 1000 像個宿主,名稱 i 寄生在它上面。C 中的 i 與 Python 中的 1000,它們則寄生在底層的記憶體空間上……

還可以這樣理解:C 中的變數 i 是一等公民,數字 1000 是它的一個可變的屬性;Python 中的數字 1000 是一等公民,名稱 i 是它的一個可變的屬性。

有了以上的鋪墊,我們再來看看 i++,不難發現:

  • C 之類的語言,i++ 可以表示 i 的數字屬性的增加,它不會開闢新的記憶體空間,也不會產生新的一等公民
  • Python 之類的語言,i++ 如果是對其名稱屬性的操作,那樣就沒有意義了(總不能按字母表順序,把 i 變成 j 吧);如果理解成對數字本體的操作,那麼情況就會變得複雜:它會產生新的一等公民 1001,因此需要給它分配一個記憶體地址,此時若占用 1000 的地址,則涉及舊對象的回收,那原有對於 1000 的引用關係都會受到影響,所以只能開闢新的記憶體空間給 1001

Python 若支持 i++,其操作過程要比 C 的 i++ 複雜,而且其含義也不再是“令數字增加1”(自增),而是“創建一個新的數字”(新增), 這樣的話,“自增操作符”(increment operator)就名不副實了。

Python 在理論上可以實現 i++ 操作,但它就必須重新定義“自增操作符”,還會令有其它語言經驗的人產生誤解,不如就讓大家直接寫成i += 1 或者 i = i + 1 好了。

2、Python 有可迭代對象

C/C++ 等語言設計出 i++,最主要的目的是為了方便使用三段式的 for 結構:

for(int i = 0; i < 100; i++){
    // 執行 xxx
}

這種程式關心的是數字本身的自增過程,數字做加法與程式體的執行相關聯。

Python 中沒有這種 for 結構的寫法,它提供了更為優雅的方式:

for i in range(100):
    # 執行 xxx

my_list = ["你好", "我是Python貓", "歡迎關註"]
for info in my_list:
    print(info)

這裡體現了不同的思維方式,它關心的是在一個數值範圍內的迭代遍歷,並不關心也不需要人為對數字做加法。

Python 中的可迭代對象/迭代器/生成器提供了非常良好的迭代/遍歷用法,能夠做到對 i++ 的完全替代。

例如,上例中實現了對列表內值的遍歷,Python 還可以用 enumerate() 實現對下標與具體值的同時遍歷:

my_list = ["你好", "我是Python貓", "歡迎關註"]
for i, info in enumerate(my_list):
    print(i, info)

# 列印結果:
0 你好
1 我是Python貓
2 歡迎關註

再例如對於字典的遍歷,Python 提供了 keys()、values()、items() 等遍歷方法,非常好用:

my_dict = {'a': '1', 'b': '2', 'c': '3'}
for key in my_dict.keys():
    print(key)

for key, value in my_dict.items():
    print(key, value)

有了這樣的利器,哪裡還有 i++ 的用武之地呢?

不僅如此,Python 中基本上很少使用i += 1 或者 i = i + 1 ,由於存在著隨處可見的可迭代對象,開發者們很容易實現對一個數值區間的操作,也就很少有對於某個數值作累加的訴求了。

所以,回到我們開頭的問題,其實這兩種“自增”寫法並沒有勝出 i++ 多少,只因為它們是通用型操作,又不需要引入新的操作符,所以 Python 才延續了一種基礎性的支持。真正的贏家其實是各種各樣的可迭代對象!

稍微小結下:Python 不支持自增操作符,一方面是因為它的整數是不可變類型的一等公民,自增操作(++)若要支持,則會帶來歧義;另一方面主要因為它有更合適的實現,即可迭代對象,對遍歷操作有很好的支持。

如果你覺得本文分析得不錯,那你應該會喜歡這些文章:

1、Python為什麼使用縮進來劃分代碼塊?

2、Python 的縮進是不是反人類的設計?

3、Python 為什麼不用分號作語句終止符?

4、Python 為什麼沒有 main 函數?為什麼我不推薦寫 main 函數?

5、Python 為什麼推薦蛇形命名法?

寫在最後:本文屬於“Python為什麼”系列(Python貓出品),該系列主要關註 Python 的語法、設計和發展等話題,以一個個“為什麼”式的問題為切入點,試著展現 Python 的迷人魅力。部分話題會推出視頻版,請在 B 站收看,觀看地址:視頻地址

公眾號【Python貓】, 本號連載優質的系列文章,有Python為什麼系列、喵星哲學貓系列、Python進階系列、好書推薦系列、技術寫作、優質英文推薦與翻譯等等,歡迎關註哦。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 屏幕解析度基礎概念說明 縮寫 全稱 說明 PX Device Pixels 設備像素,指設備的物理像素 PX CSS Pixels CSS像素,指CSS樣式代碼中使用的邏輯像素 DOT Dot 點,屏幕或列印紙上的點,等同物理像素 PT Point 點(傳統長度單位)為1/72英寸=0.35mm P ...
  • 一直以來,Web前端領域最大的問題就是相容性問題,沒有之一。 前端相容性問題分三類:瀏覽器相容性問題、屏幕解析度相容性問題、跨平臺相容性問題 Web前端框架也分三類:瀏覽器相容框架、響應式框架、跨平臺框架 ...
  • 首先我們來看看這個基礎模版的目錄結構 ├── build // 構建相關 ├── config // 配置相關├── src // 源代碼│ ├── api // 所有請求│ ├── assets // 主題 字體等靜態資源│ ├── components // 全局公用組件│ ├── direct ...
  • Git 流程圖 Workspace:工作區 Index / Stage:暫存區 Repository:倉庫區(或本地倉庫) Remote:遠程倉庫 配置 Git:另外技術是不斷更新的。在此贈送2020最新企業級別Vue3.0/Js/ES6/TS/React/node等實戰視頻教程,想學的可進裙 51 ...
  • "不畏懼,不講究,未來的日子好好努力"——大家好!我是小芝麻😄 首先感謝大家對小芝麻上一篇《想自學JS嗎?想提升JS底層原理嗎?76張腦圖帶你徹底搞懂原生JS》的支持和認可🙏,,其次是感謝大家的包容和鼓勵; 特別、特別、特別感謝!!! 另外筆者也成功升到了3級😄 ;既然大家都沒什麼要求,那本芝 ...
  • ​一、char類型 1.char類型存儲的是整數而不是字元,例如:在ASCII中,65代表A; 2.標準ASCII範圍為0-127;商用統一編碼Unicode創建一個全球字元合集; 3.註意點:C語言把1位元組定義為char類型占用的位數,因此在16位,32位,64位系統中都是可以用char類型的。有 ...
  • 1. Java線程理解 進程:進程就相當於一個應用程式,而線程是進程中的執行場景或者說執行單元,一個進程可以啟動多個線程。 線程併發:對於電腦的CPU,例如4核的CPU,表示在同一個時間點上,可以真正做到有4個進程併發執行。而對於單核CPU,是不能做到真正的多線程併發的,只是由於CPU線上程之間切換 ...
  • 操作系統 : CentOS7.7.1908_x64 gcc版本 :4.8.5 Python 版本 : 3.6.8 安裝語音識別環境: virtualenv -p /usr/bin/python3 py36asr source py36asr/bin/activate pip install Spee ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...