一篇文章掌握 Python 內置 zip() 的全部內容

来源:https://www.cnblogs.com/pythonista/archive/2020/07/09/13276597.html
-Advertisement-
Play Games

一篇文章掌握 Python 內置 zip() 的全部內容 zip() 是 Python 中最好用的內置類型之一,它可以接收多個可迭代對象參數,再返回一個迭代器,可以把不同可迭代對象的元素組合起來。 我之前寫迭代器系列的時候,在《Python進階:設計模式之迭代器模式》中簡單地介紹過它,前幾天翻譯了 ...


一篇文章掌握 Python 內置 zip() 的全部內容

zip() 是 Python 中最好用的內置類型之一,它可以接收多個可迭代對象參數,再返回一個迭代器,可以把不同可迭代對象的元素組合起來。

我之前寫迭代器系列的時候,在《Python進階:設計模式之迭代器模式》中簡單地介紹過它,前幾天翻譯了 Python 3.10 採納的 PEP-618 ,介紹了它將會迎來的變更。

但是,還有不少同學並不知道 zip(),或者不能熟練掌握它的用法,因此本文打算來做一個更為詳細的梳理。

內容主要分三部分:

  • 用法部分:介紹它的基礎用法、高級用法、騷操作用法
  • 進階部分:介紹它的實現原理,關註幾個實現的細節
  • 發散部分:聚焦它的不足,以及解決方法

1、zip() 的 n 種用法

基本用法:像拉鏈一樣,將多個可迭代對象組合起來,然後可以用 for 迴圈依次取出,或者一次性將結果存入列表、元組或者字典之類的容器中。

它的結果是一個迭代器,迭代器生成的元素是元組,第 i 個元組的元素分別來自可迭代對象參數的第 i 個元素,如上圖所示。

另外,for 迴圈還可以把元組內的元素依次取出,這樣會很方便:

它的參數並不要求是同一類的可迭代對象,因此可以有非常多的組合方式,例如:

但是,如果把字典作為 zip() 的參數,會是什麼結果呢?字典是 key-value 鍵值對形式,跟列表之類的單一元素結構不同。

實驗一下,可以看出,zip() 預設只會遍歷字典的 key 值:

如果想要取出字典的 value 值,或者取出 key-value 鍵值對,那麼可以使用字典自帶的遍歷方法 values() 和 items():

使用 zip(),還可以比較方便地對二維列表實現行列轉換:

上例中的星號(*)操作符可以解包(unpacking),即將 my_list 的元素(也是列表)解成多個參數給 zip(),從而將 3 個列表重新組合。

解包操作符對於 zip 對象同樣適用,因為 zip() 本身是一次行列轉換的操作,若將它解包後作為參數給 zip(),等於再做一次行列轉換,也就是回到了原點(除了最後的結果是元組):

最後再介紹一種用法:創建 n*n 的方陣,每行的數字相同。

2、zip() 的原理解析

官方文檔中給出了 zip() 的 Python 偽代碼(並非是 Python 解釋器內置的實現,只為了展示基本的代碼邏輯):

def zip(*iterables):
    # zip('ABCD', 'xy') --> Ax By
    sentinel = object()
    iterators = [iter(it) for it in iterables]
    while iterators:
        result = []
        for it in iterators:
            elem = next(it, sentinel)
            if elem is sentinel:
                return
            result.append(elem)
        yield tuple(result)

在這段簡短的代碼中,可以分析出幾點關鍵的信息:

  • zip 接收可變數量的可迭代對象參數,這些參數會經過 iter() 處理成迭代器。推論:若出現非可迭代對象,此處會報錯
  • while 迴圈在判斷列表是否為空,而列表中的元素是將參數轉化而成的迭代器。推論:若入參存在有效的可迭代對象,則 while 迴圈始終為真;若沒有入參,則什麼都不做
  • next() 會依次讀取迭代器中的下一個元素,它的第二個參數會作為迭代器耗盡時的返回值。推論:每一輪依次取出這些迭代器的一個元素,當某個迭代被耗盡時,則退出死迴圈,這就意味著未耗盡的迭代器會被直接捨棄

3、zip() 的問題與解決

zip() 最明顯的問題是它會捨棄掉未耗盡的迭代器:

這是一種木桶效應,最終的結果由最短的木板來決定。

有一種解決思路是取長板,同時補足短板(用 None 值填充),這就是 itertools 中的 zip_longest 方法:

它填充了冗餘數據,同時最大限度地保證了原始數據的完整性。

但是,如果我們不希望有冗餘數據,只希望得到按最長方式對齊的數據呢?

Python 官方最近採納了 PEP-618,它就是為了應對這個問題。當出現迭代器長度不一致時,它既不向短板妥協,也不向長板妥協,而是拋出 ValueError。它認為入參值錯誤,也就是嚴格要求入參的數據完整性。

該 PEP 會被合入到一年後的 Python 3.10 版本,關於更多的內容細節,可查閱這篇PEP-618 譯文


公眾號:Python貓

頭條號:Python貓

知乎:豌豆花下貓

掘金:豌豆花下貓


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

-Advertisement-
Play Games
更多相關文章
  • 昨晚安裝pymssql時,失敗了,提示pip版本需要升級: 下麵是升維pip: python -m pip install --upgrade pip。 ...
  • 九九乘法表 中文名 產生年代 別稱 最早見於 九九乘法表 春秋戰國 九九歌 《演算法大成》 /** * 九九乘法表 */ public class MultiplicationTable { public static void main(String[] args) { // 定義第一層for語句循 ...
  • pip使用 最近在學人工智慧,需要用到pip,但是當我打開時,發現出現了一系列錯誤 Unknown or unsupported command 'install' 又或者是 Did not provide a command 但是出現了這種情況後,該怎麼辦呢 在經過N次折騰之後,執行了下where ...
  • 1. 按位運算符 2. 按位運算圖解 3. 按位右移圖解 4. 按位與、按位或圖解 5. 按位異或、按位取反圖解 6. 二進位正負數轉換 7. 位運算符總結 進位轉換: 十六位數對應位置的表示方法 八進位轉二進位 十進位轉二進位 十六進位轉二進位 二進位轉八進位 二進位轉十進位 二進位轉十六進位 十 ...
  • 賦值運算符分類 基本賦值運算符:=(支持連續賦值) 擴展賦值運算符:+=,-=,*=,/=,%= 1 short s = 3; 2 s = s + 2; 3 s+=2; 上述代碼第2行和第3行有什麼區別?第2行:變數s為short類型,2預設為int類型,兩者相加後賦值給short類型的變數s時,會 ...
  • @ 前言 前面幾篇文章分析了Mybatis的核心原理,但模塊較多,沒有一一分析,更多的需要讀者自己下來研究。不過Mybatis的插件擴展機制還是非常重要的,像PageHelper就是一個擴展插件,熟悉其擴展原理,才能更好的針對我們的業務作出更合適的擴展。另外,現在Mybatis都是和Spring/S ...
  • 本文源碼:GitHub·點這裡 || GitEE·點這裡 一、分散式事務簡介 1、轉賬經典案例 跨地區和機構的轉賬的業務在實際生活中非常常見,基礎流程如下: 賬戶01通過一系列服務和支付的流程,把錢轉入賬戶02,在這一過程中,如果賬戶01出現出賬成功,但是賬戶02沒有入賬,這就導致數據不一致,違反了 ...
  • 在解釋這兩個函數之前,為了方便大家更容易理解,我們先來講講下麵的這幅圖。 ​ 這個圖中的例子其實很簡單,我們就是把題目通過輸入設備(例如鍵盤滑鼠)傳輸到電腦中,然後讓電腦進行運算得出結果,再然後把結果通過輸出設備(例如顯示器)反饋給我們,這是一個簡單的人機的交互過程。 在 C 語言中其實也是一樣 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...