python之迭代器和生成器

来源:http://www.cnblogs.com/work115/archive/2016/08/30/5822528.html
-Advertisement-
Play Games

一、迭代器(iterator) 在Python中,for迴圈可以用於Python中的任何類型,包括列表、元祖等等,實際上,for迴圈可用於任何“可迭代對象”,這其實就是迭代器 迭代器是一個實現了迭代器協議的對象,Python中的迭代器協議就是有next方法的對象會前進到下一結果,而在一系列結果的末尾 ...


一、迭代器(iterator)

在Python中,for迴圈可以用於Python中的任何類型,包括列表、元祖等等,實際上,for迴圈可用於任何“可迭代對象”,這其實就是迭代器

迭代器是一個實現了迭代器協議的對象,Python中的迭代器協議就是有next方法的對象會前進到下一結果,而在一系列結果的末尾是,則會引發StopIteration。任何這類的對象在Python中都可以用for迴圈或其他遍歷工具迭代,迭代工具內部會在每次迭代時調用next方法,並且捕捉StopIteration異常來確定何時離開。

使用迭代器一個顯而易見的好處就是:每次只從對象中讀取一條數據,不會造成記憶體的過大開銷。

比如要逐行讀取一個文件的內容,利用readlines()方法,我們可以這麼寫:

1
2
for line in open("test.txt").readlines():
print line

這樣雖然可以工作,但不是最好的方法。因為他實際上是把文件一次載入到記憶體中,然後逐行列印。當文件很大時,這個方法的記憶體開銷就很大了。

利用file的迭代器,我們可以這樣寫:

1
2
for line in open("test.txt"):   #use file iterators
print line

這是最簡單也是運行速度最快的寫法,他並沒顯式的讀取文件,而是利用迭代器每次讀取下一行。

二、生成器(constructor)

生成器函數在Python中與迭代器協議的概念聯繫在一起。簡而言之,包含yield語句的函數會被特地編譯成生成器。當函數被調用時,他們返回一個生成器對象,這個對象支持迭代器介面。函數也許會有個return語句,但它的作用是用來yield產生值的。

不像一般的函數會生成值後退出,生成器函數在生成值後會自動掛起並暫停他們的執行和狀態,他的本地變數將保存狀態信息,這些信息在函數恢復時將再度有效

1
2
3
4
5
6
7
8
>>> def g(n):
... for i in range(n):
... yield i **2
...
>>> for i in g(5):
... print i,":",
...
0 : 1 : 4 : 9 : 16 :

要瞭解他的運行原理,我們來用next方法看看:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
>>> t = g(5)
>>> t.next()
0
>>> t.next()
1
>>> t.next()
4
>>> t.next()
9
>>> t.next()
16
>>> t.next()
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
StopIteration

在運行完5次next之後,生成器拋出了一個StopIteration異常,迭代終止。
再來看一個yield的例子,用生成器生成一個Fibonacci數列:

1
2
3
4
5
6
7
8
9
10
def fab(max):
a,b = 0,1
while a < max:
yield a
a, b = b, a+b
 
>>> for i in fab(20):
... print i,",",
...
0 , 1 , 1 , 2 , 3 , 5 , 8 , 13 ,

看到這裡應該就能理解生成器那個很抽象的概念了吧~~

 def read_file(fpath): 
    BLOCK_SIZE = 1024 
    with open(fpath, 'rb') as f: 
        while True: 
            block = f.read(BLOCK_SIZE) 
            if block: 
                yield block 
            else: 
                return
複製代碼

如果直接對文件對象調用 read() 方法,會導致不可預測的記憶體占用。好的方法是利用固定長度的緩衝區來不斷讀取文件內容。通過 yield,我們不再需要編寫讀文件的迭代類,就可以輕鬆實現文件讀取。


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

-Advertisement-
Play Games
更多相關文章
  • 1、加密演算法 為了網路通訊中的報文安全,一般需要對報文進行加密,目前常用的加密演算法有: 非對稱加密演算法:又稱公鑰加密演算法,如RSA、DSA/DSS,最常用的就是RSA演算法(演算法公開,可自行百度瞭解演算法細節),演算法產生一個公鑰一個私鑰,用公鑰加密的報 文只能用私鑰解密,用私鑰加密的報文只能用公鑰解密; ...
  • 對於那些創建耗時較長,或者資源占用較多的對象,比如網路連接,線程之類的資源,通常使用池化來管理這些對象,從而達到提高性能的目的。比如資料庫連接池(c3p0, dbcp), java的線程池 ExecutorService.Apache Commons Pool提供一套池化規範介面,以及實現通用邏輯, ...
  • 1 下載golang安裝包和配置環境變數 到官網下載golang安裝包,下載地址:https://golang.org/dl/,我選擇的go1.7.windows-amd64.zip. 配置環境變數:GOROOT和GOBIN 2 下載notepad++和GOnpp插件 notepad++下載地址:h ...
  • SpringMVC原理圖: 步驟: 前端控制器對視圖進行渲染。 View——>渲染,View會根據傳進來的Model模型數據進行渲染,此處的Model實際是一個Map數據結構,所以其很容易支持其他視圖技術。 組件介紹: 前端控制器(DispatcherServlet):接收請求,響應結果。 處理器映 ...
  • 目標:利用R語言統計描繪50組實驗對比結果 第一步:導入.csv文件 X <- read.table("D:abc11.csv",header = TRUE, sep = ",") 第二步:繪圖 ggplot(X, aes(x = aaa, y = bbb)) + geom_point() + ge ...
  • 一、面向過程 VS 面向對象 編程範式 編程是程式員用特定的語法+數據結構+演算法組成的代碼來告訴電腦如何執行任務的過程 ,一個程式是程式員為了得到一個任務結果而編寫的一組指令的集合,正所謂條條大路通羅馬,實現一個任務的方式有很多種不同的方式,對這些不同的編程方式的特點進行歸納總結得出來的編程方式類 ...
  • Java 同步塊(synchronized block)用來標記方法或者代碼塊是同步的。Java 同步塊用來避免競爭。本文介紹以下內容: Java 同步關鍵字(synchronzied) 實例方法同步 靜態方法同步 實例方法中同步塊 靜態方法中同步塊 Java 同步示例 Java 同步關鍵字(syn ...
  • 1 Servlet生命周期Servlet 生命周期:Servlet 載入 >實例化 >服務 >銷毀。 init():在Servlet的生命周期中,僅執行一次init()方法。它是在伺服器裝入Servlet時執行的,負責初始化Servlet對象。可以配置伺服器,以在啟動伺服器或客戶機首次訪問Servl ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...