Python 容器使用的 5 個技巧和 2 個誤區

来源:https://www.cnblogs.com/xxpythonxx/archive/2019/09/23/11575081.html
-Advertisement-
Play Games

“容器”這兩個字很少被 Python 技術文章提起。一看到“容器”,大家想到的多是那頭藍色小鯨魚:Docker,但這篇文章和它沒有任何關係。本文里的容器,是 Python 中的一個抽象概念,是對專門用來裝其他對象的數據類型的統稱。 在 Python 中,有四類最常見的內建容器類型: 列表(list) ...


“容器”這兩個字很少被 Python 技術文章提起。一看到“容器”,大家想到的多是那頭藍色小鯨魚:Docker,但這篇文章和它沒有任何關係。本文里的容器,是 Python 中的一個抽象概念,是對專門用來裝其他對象的數據類型的統稱。

在 Python 中,有四類最常見的內建容器類型: 列表(list)、 元組(tuple)、 字典(dict)、 集合(set)。通過單獨或是組合使用它們,可以高效的完成很多事情。

Python 語言自身的內部實現細節也與這些容器類型息息相關。比如 Python 的類實例屬性、全局變數 globals() 等就都是通過字典類型來存儲的。

在這篇文章里,我首先會從容器類型的定義出發,嘗試總結出一些日常編碼的最佳實踐。之後再圍繞各個容器類型提供的特殊機能,分享一些編程的小技巧。

當我們談論容器時,我們在談些什麼?

我在前面給了“容器”一個簡單的定義:專門用來裝其他對象的就是容器。但這個定義太寬泛了,無法對我們的日常編程產生什麼指導價值。要真正掌握 Python 里的容器,需要分別從兩個層面入手:

  • 底層實現:內置容器類型使用了什麼數據結構?某項操作如何工作?
  • 高層抽象:什麼決定了某個對象是不是容器?哪些行為定義了容器?

下麵,讓我們一起站在這兩個不同的層面上,重新認識容器。

底層看容器

Python 是一門高級編程語言,它所提供的內置容器類型,都是經過高度封裝和抽象後的結果。和“鏈表”、“紅黑樹”、“哈希表”這些名字相比,所有 Python 內建類型的名字,都只描述了這個類型的功能特點,其他人完全沒法只通過這些名字瞭解它們的哪怕一丁點內部細節。

這是 Python 編程語言的優勢之一。相比 C 語言這類更接近電腦底層的編程語言,Python 重新設計並實現了對編程者更友好的內置容器類型,屏蔽掉了記憶體管理等額外工作。為我們提供了更好的開發體驗。

但如果這是 Python 語言的優勢的話,為什麼我們還要費勁去瞭解容器類型的實現細節呢?答案是:關註細節可以幫助我們編寫出更快的代碼。

寫更快的代碼

1. 避免頻繁擴充列表/創建新列表

所有的內建容器類型都不限制容量。如果你願意,你可以把遞增的數字不斷塞進一個空列表,最終撐爆整台機器的記憶體。

在 Python 語言的實現細節里,列表的記憶體是按需分配的[註1],當某個列表當前擁有的記憶體不夠時,便會觸發記憶體擴容邏輯。而分配記憶體是一項昂貴的操作。雖然大部分情況下,它不會對你的程式性能產生什麼嚴重的影響。但是當你處理的數據量特別大時,很容易因為記憶體分配拖累整個程式的性能。

還好,Python 早就意識到了這個問題,並提供了官方的問題解決指引,那就是:“變懶”。

如何解釋“變懶”? range() 函數的進化是一個非常好的例子。

在 Python 2 中,如果你調用 range(100000000),需要等待好幾秒才能拿到結果,因為它需要返回一個巨大的列表,花費了非常多的時間在記憶體分配與計算上。但在 Python 3 中,同樣的調用馬上就能拿到結果。因為函數返回的不再是列表,而是一個類型為 range 的懶惰對象,只有在你迭代它、或是對它進行切片時,它才會返回真正的數字給你。

所以說,為了提高性能,內建函數 range “變懶”了。而為了避免過於頻繁的記憶體分配,在日常編碼中,我們的函數同樣也需要變懶,這包括:

  • 更多的使用 yield 關鍵字,返回生成器對象
  • 儘量使用生成器表達式替代列表推導表達式
    • 生成器表達式: (iforinrange(100))
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在上篇文章: "SpringBoot源碼解析:創建SpringApplication對象實例" 中,我們詳細描述了SpringApplication對象實例的創建過程,本篇文章繼續看 方法的執行邏輯吧 1. 第一行使用了 來記錄開始時間 2. 設置了 環境變數,在網上瞭解了一下這個變數的相關信息 H ...
  • [TOC] 1. 數組操作符重載 數組操作符重載 通過重載數組操作符,可以使類的對象支持數組的下標訪問 數組操作符只能重載為類的成員函數 重載函數能且僅能使用一個參數,也就是數組下標 可以定義不同參數的多個重載函數 在重載數組操作符時,要記得數組操作符的原生語義——數組訪問和指針運算。 cpp / ...
  • 從今天起,我會在這裡記錄一下學習深度學習所留下的足跡,目的也很簡單,手頭有近3w個已經標記好正確值得驗證碼,想要從頭訓練出一個可以使用的模型, 雖然我也知道網上的相關模型和demo很多,但是還是非常希望自己可以親手搞一個能用的出來,學習書籍主要是:李金洪老師的《深度學習之Tensorflow 入門、 ...
  • 在做數據分析的過程中,經常會遇到文件的讀取。我想很多人都在這個環節遇到過問題,所以就把自己掌握的一些文件讀取方法記錄下來,以及過程中遇到的一些狀況和解決方法列出來,以便交流。 open open() 函數用於創建或打開指定文件,該函數的語法格式如下: 參數說明: file:表示要創建的文件對象。 f ...
  • 實在不想看JVM了。刷幾道劍指Offer的題,今天就水一水吧,腦子迷糊。 1.二維數組中的查找 在一個二維數組中(每個一維數組的長度相同),每一行都按照從左到右遞增的順序排序,每一列都按照從上到下遞增的順序排序。請完成一個函數,輸入這樣的一個二維數組和一個整數,判斷數組中是否含有該整數。 解題思路: ...
  • 一、題目 二、思路 1、dfs 實驗要求用多種思路完成,所以一開始就沿用了上一個實驗馬走棋盤的思路,添加了鄰接矩陣來記錄有向網的權值。總體思路還是DFS遍歷搜索。 過程剪枝: 1、因為要求為最短路徑,而一般情況總會存在多條可行路徑,在判斷過程中需要走過每一條路徑才能知道該路徑的長度,但如果已知一條可 ...
  • [TOC] 閉包函數 什麼是閉包函數 閉包函數把 閉包函數內的變數 + 閉包函數內部的函數, 這兩者包裹起來,然後通過返回值的形式返回出來。 定義在函數的內函數 該函數體代碼包含對該函數外層作用域中變數的引用 函數外層指的不是全局作用域 上述代碼中,f是一個全局的名字,但f拿到了inner的記憶體地址 ...
  • 我是一個2019畢業的非電腦的畢業生,從大二開始喜歡上Java直到現在一直都在學習,Brid從小就對電腦感興趣,可惜高中的時候不懂事,沒有規劃未來,考上了一所專科學院,然後大一併不能轉專業,現在畢業了沒有找到Java應屆的工作,只能找點其他的做,但是這阻住不了我對Java的喜歡,趁現在工作的晚上 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...