python學習筆記:第11天 閉包及迭代器

来源:https://www.cnblogs.com/zpzhue1/archive/2018/10/31/9885981.html
-Advertisement-
Play Games

[TOC] 1. 函數名的使用 其實函數名也是一個變數,但它是一個比較特殊的變數,與小括弧配合可以執行函數的變數: 函數名其實和記憶體一樣,也可以使用 查看它的記憶體地址: 函數名賦值給其他變數 函數也能當作容器類的元素: 函數名也能當作函數的參數: 函數名也可以作為函數的返回值: 2. 閉包 閉包是指 ...


目錄

1. 函數名的使用

其實函數名也是一個變數,但它是一個比較特殊的變數,與小括弧配合可以執行函數的變數:

  • 函數名其實和記憶體一樣,也可以使用print查看它的記憶體地址:
In[22]: def func1():
   ...:     pass
   ...: 
In[23]: print(func1)
<function func1 at 0x000002A24830C488>
  • 函數名賦值給其他變數
In[24]: def func2():
   ...:     print('呵呵')
   ...:     
In[25]: a = func2
In[26]: a()
呵呵
  • 函數也能當作容器類的元素:
In[27]: print(func2)
<function func2 at 0x000002A24830B048>
In[28]: lis = [func2, func2, func2]
In[29]: print(lis)
[<function func2 at 0x000002A24830B048>, <function func2 at 0x000002A24830B048>, <function func2 at 0x000002A24830B048>]
  • 函數名也能當作函數的參數:
In[30]: def func3():
   ...:     print('i\'m func3')
   ...:     
In[31]: def func4(fn):
   ...:     fn()
   ...:     
In[32]: func4(func3)        # 把函數名func3作為參數傳遞給func4
i'm func3
  • 函數名也可以作為函數的返回值:
In[33]: def func5():
   ...:     def func6():
   ...:         print('this is sub function')
   ...:     return func6        # 這裡直接把函數名func6作為返回值返回給調用者
   ...: 
In[34]: fn = func5()            # 這裡的fn就是func6了
In[35]: fn()                    # 加()執行函數
this is sub function

2. 閉包

閉包是指在內層函數中對外層函數(非全局)的引用

In[36]: def func6():
   ...:     x = 24
   ...:     def func7():
   ...:         print(x)                # 閉包
   ...:     func7()
   ...:     print(func7.__closure__)    # 使⽤__closure__來檢測函數是否是閉包. 
   ...:                                 # 使⽤函數名.__closure__返回cell就是閉包. 返回None就不是閉包
     
In[37]: func6()
24
(<cell at 0x000002A2482F8EE8: int object at 0x000000005BA86F00>,)

那麼我們要怎麼能在函數外面調用內部函數呢,其實很簡單,把內部函數作為返回值返回給調用者即可:

In[38]: def func8():
   ...:     x = 24
   ...:     def func9():
   ...:         print(x)
   ...:     print(func9.__closure__)
   ...:     return func9    
   ...: 
In[39]: fn = func8()
(<cell at 0x000002A2482F8DF8: int object at 0x000000005BA86F00>,)
In[40]: fn()                            # 這樣就可以在函數外面使用了
24

那麼閉包有什麼用呢,我們再來看一個例子:

In[2]: def func1():
  ...:     x = 23
  ...:     def func2():
  ...:         nonlocal x
  ...:         x += 1
  ...:         return x
  ...:     return func2
  ...: 
In[3]: fn = func1()
In[5]: fn()
Out[5]: 24
In[6]: fn()
Out[6]: 25
In[7]: fn()
Out[7]: 26
In[8]: fn()
Out[8]: 27
In[9]: x
Traceback (most recent call last):
  File "D:\Environment\python-virtualenv\jupyter\lib\site-packages\IPython\core\interactiveshell.py", line 3265, in run_code
    exec(code_obj, self.user_global_ns, self.user_ns)
  File "<ipython-input-9-6fcf9dfbd479>", line 1, in <module>
    x
NameError: name 'x' is not defined

從上面我們可以看出,x作為一個局部命名空間的變數,在使用是看起來更像是使用全局變數一樣,但是最後的報錯是證明瞭x並不是一個全局變數。這個現象就是閉包造成的,它可以把函數中的變數在外部使用,並且能讓它常駐於記憶體。

3. 迭代器

我們之前使用for迴圈變數一個容器類的對象是,都有提要遍歷的對象一定是要可迭代的,先看下可迭代對象裡面都有什麼:

In[15]: dir(list)
Out[15]: 
['__add__',
 '__class__',
...
...
 '__imul__',
 '__init__',
 '__init_subclass__',
 '__iter__',            # 列表這裡有個__iter__方法,代表這個是一個可迭代的對象
 '__le__',
 '__len__',
 '__lt__',
 '__mul__',
...
...
In[16]: dir(str)
Out[16]: 
['__add__',
 '__class__',
 ...
 ...
 '__init__',
 '__init_subclass__',
 '__iter__',            # 字元串也是有__iter__方法
 '__le__',
 '__len__',
 ...
 ...

如果自己嘗試過的話會發現列表、字典、字元串和集合都會有這個方法,因為他們都是可迭代對象。

這是查看⼀個對象是否是可迭代對象的第⼀種辦法. 我們還可以通過isinstence()函數來查看⼀個對象是什麼類型的

l = [1,2,3]
l_iter = l.__iter__()
from collections import Iterable
from collections import Iterator
print(isinstance(l,Iterable))           #True
print(isinstance(l,Iterator))           #False
print(isinstance(l_iter,Iterator))      #True
print(isinstance(l_iter,Iterable))      #True

綜上. 我們可以確定. 如果對象中有__iter__函數. 那麼我們認為這個對象遵守了可迭代協議.就可以獲取到相應的迭代器. 這⾥的__iter__是幫助我們獲取到對象的迭代器. 我們使⽤迭代器中的__next__()來獲取到⼀個迭代器中的元素. 那麼我們之前講的for的⼯作原理到底是什麼? 繼續看代碼

In[17]: s = 'zzc'
In[18]: s_iter = s.__iter__()       # 使用字元串的__iter__()方法
In[19]: s_iter.__next__()
Out[19]: 'z'
In[20]: s_iter.__next__()
Out[20]: 'z'
In[21]: s_iter.__next__()
Out[21]: 'c'
In[22]: s_iter.__next__()
Traceback (most recent call last):
  File "D:\Environment\python-virtualenv\jupyter\lib\site-packages\IPython\core\interactiveshell.py", line 3265, in run_code
    exec(code_obj, self.user_global_ns, self.user_ns)
  File "<ipython-input-22-b111e2554a10>", line 1, in <module>
    s_iter.__next__()
StopIteration

從上可以看出,只要一個對象有__iter__方法,那麼我們認為這個對象遵守了可迭代協議,就可以獲取到相應的迭代器(s_iter),然後後我們可以使用迭代器中的__netx__方法來獲取下一個迭代器中的元素,直到拋出`StopIteration``異常時退出,

for迴圈的機制:

In[24]: l1 = ['zzc', '牛奶', 'PDD', '55開']
In[25]: for i in l1:
   ...:     print(i)
   ...:     
zzc
牛奶
PDD
55開

用while實現的for迴圈:

lis = ['zzc', '牛奶', 'PDD', '55開']
iter = lis.__iter__()
while 1:
    try:                        # try/excpet是捕獲異常的語句
        ele = iter.__next__()
        print(ele)
    except StopIteration:       # 當捕獲到StopIteration異常時退出
        break

總結:

  • Iterable: 可迭代對象. 內部包含__iter__()函數
  • Iterator: 迭代器. 內部包含__iter__() 同時包含__next__().
  • 迭代器的特點:
    1. 節省記憶體.
    2. 惰性機制
    3. 不能反覆, 只能向下執⾏.

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

-Advertisement-
Play Games
更多相關文章
  • 1. fgetss函數php官網的解釋是: (PHP 4, PHP 5, PHP 7) fgetss — 從文件指針中讀取一行並過濾掉 HTML 標記 2. 測試後出現的問題是: 當文本中有一行數據出現 < 左尖括弧字元時,會把下麵的數據全部替換成空白行 ,每行讀取到的數據都是空白 ...
  • 前面提到條件語句的標準格式為“if (條件) { /* 條件成立時的操作代碼 */ } else { /* 條件不成立時的操作代碼 */ }”,乍看之下仿佛只有兩個分支,一個是條件成立時的分支,另一個是條件不成立時的分支。很明顯僅僅兩個分支是不能滿足複雜的業務需求的,自然Java代碼也不會這麼傻瓜到 ...
  • 1、使用IDEA新建工程引導方式,創建消息生產工程 springboot-kafka-producer。 工程POM文件代碼如下: 註釋部分為手動添加的 gson、lombok 依賴。 2、創建消息實體類 3、創建消息生產類 4、編輯資源配置文件 application.properties 5、啟 ...
  • 創建Django項目: 命令行創建:python manage.py startproject 項目名 啟動Django項目: 根目錄下(有manage.py的目錄) python manage.py runserver IP:埠(可直接寫埠或預設在本地的8000埠下) 創建APP: 命令行創 ...
  • netty 與 webSocket 起因 有個需求需要用到 ,然後最近又正好在學 ,然後合起來走一波。寫篇文章記錄一下,做一個念想。 協議格式 開始 我們先寫一個什麼都不加的 熱熱手,話不多說,代碼如下 常規的netty入門示例,加了個String的編碼和解碼器,還加了一個列印消息的 ,並不是什麼太 ...
  • 1. 文件操作 open() 文件句柄 open()打開一個文件, 獲取的是文件句柄 read() #讀取全部內容 read(n)#讀取前n個字元 readline()#讀取一行 且讀取出來末尾都有\n readlines()#讀取全部 每一⾏形成一個元素並放到列表 註意: 讀取完的文件句柄一定要關 ...
  • package zrs; public class javaDay02_3 { public static void main(String[] args){ //switch 結構 int x=5; switch(x){//x 支持byte short int char 5.0以後 enum St ...
  • ·字元串(string) @ title @ title @ 小結: ·字元串(string) @ title @ title @ 小結: ****** 幾米花的Python ****** 博客主頁:https://www.cnblogs.com/jimmy-share/ 歡迎轉載 ~ ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...