Python學習日記(十) 生成器和迭代器

来源:https://www.cnblogs.com/Fantac/archive/2019/08/13/11343409.html
-Advertisement-
Play Games

使用dir()我們可以知道這個數據類型的內置函數有什麼方法: 1.迭代器 iterable:可迭代的 迭代就是將數據能夠一個一個按順序取出來 上面數據類型返回為真說明它是可以迭代的,反之是不可迭代的 可迭代協議: 就是內部要有一個__iter__()來滿足要求 當一個具有可迭代的數據執行__iter ...


使用dir()我們可以知道這個數據類型的內置函數有什麼方法:

print(dir(int))  
print(dir(bool))
print(dir([]))
print(dir({}))
print(dir(set))

 

1.迭代器

iterable:可迭代的

迭代就是將數據能夠一個一個按順序取出來

s = 'abc'
print('__iter__' in dir(s))      #True
li = [1,2]
print('__iter__' in dir(li))     #True
b = False
print('__iter__' in dir(b))      #False
i = 123
print('__iter__' in dir(i))      #False
dic = {}
print('__iter__' in dir(dic))    #True
set1 = set()
print('__iter__' in dir(set1))   #True

上面數據類型返回為真說明它是可以迭代的,反之是不可迭代的

可迭代協議:

就是內部要有一個__iter__()來滿足要求

當一個具有可迭代的數據執行__iter__()它將返回一個迭代器的記憶體地址

print('abc'.__iter__()) #<str_iterator object at 0x005401F0>

這裡的iterator的意思是迭代器

迭代器協議:

現在有一個列表我們來看看它本身和在執行了__iter__()之後的方法有什麼不同:

li = [1,2,3,'a']
print(dir(li))  #['__add__', '__class__', '__contains__', '__delattr__', '__delitem__', '__dir__', '__doc__', '__eq__', '__format__', '__ge__', '__getattribute__', '__getitem__', '__gt__', '__hash__', '__iadd__', '__imul__', '__init__', '__iter__', '__le__', '__len__', '__lt__', '__mul__', '__ne__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__reversed__', '__rmul__', '__setattr__', '__setitem__', '__sizeof__', '__str__', '__subclasshook__', 'append', 'clear', 'copy', 'count', 'extend', 'index', 'insert', 'pop', 'remove', 'reverse', 'sort']
print(dir(li.__iter__()))   #['__class__', '__delattr__', '__dir__', '__doc__', '__eq__', '__format__', '__ge__', '__getattribute__', '__gt__', '__hash__', '__init__', '__iter__', '__le__', '__length_hint__', '__lt__', '__ne__', '__new__', '__next__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__setstate__', '__sizeof__', '__str__', '__subclasshook__']
'''求兩者的差集'''
print(set(dir(li.__iter__())) - set(dir(li)))   #{'__length_hint__', '__setstate__', '__next__'}
__length_hint__()的作用是求元素的長度
__setstate__()的作用是指定索引值從哪裡開始迭代
__next__()的作用可以讓值一個一個的取出

在之前用到的for迴圈我們就是用__next__()這種方法進行取值,現在我們可以模擬for來寫一個函數:
li = [1,2,3,'a']
def getItem(li):
    iterator = li.__iter__()
    while True:
         print(iterator.__next__())
getItem(li)
# 1
# 2
# 3
# a
#StopIteration 如果找不到元素就會報錯

如何處理掉這個異常?

li = [1,2,3,'a']
def getItem(li):
    iterator = li.__iter__()
    while True:
        try:
            print(iterator.__next__())
        except StopIteration:
            break
getItem(li)
# 1
# 2
# 3
# a

迭代器遵循迭代器協議:必須要有__iter__()和__next__()

迭代器的好處:

  a.從容器類型中一個一個的取值,會把所有的值都取到

  b.節省記憶體的空間

     迭代器並不會在記憶體中占用一大塊記憶體,而是隨著迴圈每次生成一個,每一次使用__next__()來給值

range():

print(range(10000000))  #range(0, 10000000)

實際上range()在調用的時候並沒有真正生成這麼多的值,如果真的生成的話那麼記憶體可能會溢出

print('__iter__' in dir(range(10)))     #True   
print('__next__' in dir(range(10)))     #False
from collections import Iterable
from collections import Iterator
print(isinstance(range(10),Iterable))    #True  是一個可迭代對象
print(isinstance(range(10),Iterator))    #False 在執行後得到的結果並不是一個迭代器

迭代器總結:

1.可以被for迴圈的都是可迭代的

2.可迭代的內部都有__iter__()方法

3.只要是迭代器一定可以迭代

4.可迭代的變數.__iter__()方法可以得到一個迭代器

5.迭代器中的__next__()方法可以一個一個的獲取值

6.for迴圈實際上就是在使用迭代器

 

2.生成器

生成器函數

本質上就是我們自己寫的函數,只要含有yield關鍵字的函數就是生成器函數,yield不能和return共用且需要寫在函數內部

def generator():        #生成器函數
    print('a')
    yield 5
ret = generator()       #ret是一個生成器
print(ret)              #<generator object generator at 0x00503F00>

生成器函數每次執行的時候會得到一個生成器作為返回值

如果要返回函數值:

def generator():        #生成器函數
    print('a')
    yield 5
    print('b')
    yield 4
g = generator()        #g是一個生成器
print(g)               #<generator object generator at 0x00503F00>
ret = g.__next__()
print(ret)              #a
                        #5
print('---------')
ret = g.__next__()
print(ret)              #b
                        #4

執行順序:

 

使用for迴圈遍歷生成器:

def generator():        #生成器函數
    print('a')
    yield 5
    print('b')
    yield 4
    print('c')
    yield 6
g = generator()       #g是一個生成器
for i in g:
    print(i)    
# a
# 5
# b
# 4
# c
# 6

監聽文件例子:

def tail(filename):
    f = open(filename,encoding='utf-8')
    while True:
        line = f.readline()
        if line.strip():
             yield line.strip()
g = tail('tail')
for i in g:
    if 'python' in i:
        print('******',i)   
# ****** python
# ****** asd python

 

 

 

 

 


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

-Advertisement-
Play Games
更多相關文章
  • 前言 我們已經學習了單一職責原則,依賴倒置原則,介面隔離原則,李氏替換原則。可以說前面幾個原則都是為了開閉原則奠定基礎。 我們寫的程式由於實際的情況可以一定程度上違背各種設計原則。但是,開閉原則我認為作為一個程式猿無論什麼時候都需要遵循他,切記不可違背她。 基本介紹 1. 開閉原則(Open Clo ...
  • 簡單來說,通過複製的方式創建對象。 【舉個慄子】:點外賣的收貨地址 ...
  • 移動電商平臺彈性架構案例雲服務彈性機房今天先到這兒,希望對技術領導力, 企業管理,系統架構設計與評估,團隊管理, 項目管理, 產品管理,團隊建設 有參考作用 , 您可能感興趣的文章: 領導人怎樣帶領好團隊構建創業公司突擊小團隊國際化環境下系統架構演化微服務架構設計視頻直播平臺的系統架構演化微服務與D... ...
  • 第一章 概述 1 spring 以 ioc 和 aop 為內核,提供了展現層 springMVC、持久層SpringJDBC及業務層事務管理等一站式企業級應用技術。 2spring的特性 方便解耦,簡化開發。通過IOC容器,用戶可以將對象之間的依賴關係交由spring進行控制,避免硬編碼所造成的的過 ...
  • JRebel 使用 JRebel 可以在修改代碼後,動態重新載入修改的代碼,免去了代碼工程全量重建、重啟的耗時流程,有效地提高開發者的效率。在 IDEA 的插件市場搜索 JRebel for IntelliJ 找到安裝即可。 JRebel for IntelliJ 版本:2019.1.4 1、啟用自 ...
  • Problem Description Problem Description Euler is a well-known matematician, and, among many other things, he discovered that the formulan^{2} + n + 41 ...
  • 一、多線程 1.我們的環境 (1)xubuntu 16.04(2)anaconda(3)pycharm(4)python 3.6 2.程式:一堆代碼以文本的形式存入一個文檔 3.進程:程式運行的一個狀態。 特點:(1)其中包含地址控制項、記憶體、數據棧等;(2)每個進程由自己完全獨立的運行環境,多進程共 ...
  • Windows下Beyond Compare 4 30天評估到期了的話,可以嘗試下麵兩種方式: 破解方式把Beyond Compare 4安裝文件夾下麵的BCUnrar.dll文件刪掉就行了,但是這種依然會提示在試用期 BC4註冊碼:可以用下麵這個註冊碼,有效期是到2019年12月 BEGIN LI ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...