python函數知識七 閉包、裝飾器一(入門)

来源:https://www.cnblogs.com/Onlywang/archive/2019/07/24/11241567.html
-Advertisement-
Play Games

21.閉包 1. 閉包:在嵌套函數內,使用非全局變數(且不使用本層變數) 2. 閉包的作用:1.保證數據的安全性(純潔度)。2.裝飾器使用 3. ._\_closure\_\_判斷是否是閉包 22.裝飾器一(入門) 1.一個裝飾器裝飾多個函數 開放封閉原則:擴展是開放的(增加新功能),源碼是封閉的( ...


21.閉包

  1. 閉包:在嵌套函數內,使用非全局變數(且不使用本層變數)
  2. 閉包的作用:1.保證數據的安全性(純潔度)。2.裝飾器使用
  3. .__closure__判斷是否是閉包
def func():
    a = 1 
    def foo():
        print(a)
    return foo
ret = func()
ret()
#例子
def func():
    avg_lst = []
    def foo(pirce):
        avg_lst.append(pirce)
        avg = sum(avg_lst) / len(avg_lst)
        return avg
    return foo
ret = func()
print(ret(1500))
print(ret(1600))
print(ret.__closure__)#結果:(<cell at 0x000002095400B558: list object at 0x00000209540A5B48>,)證明是閉包
print(func.__closure__)#結果:None,不是閉包

print(ret.__code__.co_freevars)#獲取的是自由變數
print(ret.__code__.co_varnames)#獲取的是局部變數

22.裝飾器一(入門)

1.一個裝飾器裝飾多個函數

開放封閉原則:擴展是開放的(增加新功能),源碼是封閉的(修改已經實現的功能)

作用:在不改變源代碼及調用方式的基礎下額外增加新功能。

裝飾器:用來裝飾的工具

2.版一:
import time
start_time = time.time
def func():
    time.sleep(2)#睡眠s模擬網路延時
    print("我要飛")
func()   
print(time.time - start_time)
#改變了源代碼
3.版二:
def times(f):
    start_time = time.time()
    f()
    print(time.time() - start_time)
def foo():
    time.sleep(3)
    print("我飛的比你高")
#times(foo)#改變了調用方式
s = foo
foo = times
foo(s)#不改變調用方式
4.版三:(low版裝飾器)
def times(f):
    def inner():
        start_time = time.time()
        f()
        print(time.time() - start_time)
    return inner
def foo():
    time.sleep(3)
    print("我飛的比你高")
foo = times(foo)
foo()
5.版四:
def wrapper(f):
    def inner(a):
        start_time = time.time()
        f(a)
        print(time.time() - start_time)
    return inner#切記不加括弧

def func(a):
    print(f"{a}你不行")
func = wrapper(func)
func("alex")
#傳輸多個數據,用*args,**kwargs
6.版五(標準版裝飾器):

@wrapper#語法糖:必須放在要裝飾的函數的正上方

def wrapper(f):#f是要被裝飾的函數名
    def inner(*args,**kwargs):
        "被裝飾前"
        start_time = time.time()
        ret = f(*args,**kwargs)
        print(time.time() - start_time)
        "被裝飾後"
        return ret
    return inner#切記不加括弧
@wrapper#語法糖 -->func = wrapper(func)
def func(*args,**kwargs):
    print(f"{a}你不行")
    return "我可以返回了"
#func = wrapper(func)#有語法糖不用多次賦值
func("alex")

23.裝飾器二(進階)

1.有參裝飾器:

​ 有參裝飾器:在基礎裝飾器的基礎上再套一層函數

#有參裝飾器實現登陸驗證
msg = """
QQ
微信
抖音
請輸入您要登錄的的app:
"""
chose = input(msg).upper()
dict_flag = {'username':None,'flag':False}
def auth(a):
    def wrapper(f):
        def inner(*args,**kwargs):
            if dic_flag['flag']:
                fun(*args,**kwargs)
            else:
                if argv =="QQ":
                    print("歡迎登陸QQ")
                    user = intput("user:")
                    pwd = input("passwd:")
                    if user == "alex" and pwd == "alex123":
                        dict_flag["flag"] = True
                        dict_flag["username"] = user
                        foo(*args,**kwargs)
                        
                    else:
                        print("用戶名或密碼錯誤!")
        return inner
    return wrapper
@auth(chose)
"""
語法糖拆分:
wrapper = auth(chose)
foo = wrapper(foo)
"""
def foo():
    print("這是被裝飾的函數")
foo()

2.多個裝飾器裝飾一個函數:

​ 當被裝飾的函數正上方有多個裝飾器,先執行里被裝飾函數最近的裝飾器(小技巧:進入裝飾器,從上往下,走到最後一個裝飾器,執行被裝飾的函數,退出裝飾器從下往上)

def wrapper1(f):
    def inner1(*args,**kwargs):
        print("這是第一個裝飾器開始")
        f(*args,**kwargs)
        print("這是第一個裝飾器結束")
        return 
    return inner1

def wrapper2(f):
    def inner2(*args,**kwargs):
        print("這是第二個裝飾器開始")
        f(*args,**kwargs)
        print("這是第二個裝飾器結束")
        return 
    return inner2

@wrapper1
@wrapper2
"""
兩個語法糖等價於:
foo = wrapper2(foo)#foo == inner2
foo = wrapper1(foo)#foo = wrapper(inner2),foo == inner1
foo()#-->inner1()
"""
def foo():
    print("這是被裝飾的函數")
foo()
'''
結果:
這是第一個裝飾器開始
這是第二個裝飾器開始
這是被裝飾的函數
這是第二個裝飾器結束
這是第一個裝飾器結束
'''


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

-Advertisement-
Play Games
更多相關文章
  • 背景 在企業發展初期,企業使用的系統很少,通常一個或者兩個,每個系統都有自己的登錄模塊,運營人員每天用自己的賬號登錄,很方便。 但隨著企業的發展,用到的系統隨之增多,運營人員在操作不同的系統時,需要多次登錄,而且每個系統的賬號都不一樣,這對於運營人員 來說,很不方便。於是,就想到是不是可以在一個系統 ...
  • 設計模式是對軟體設計中普遍存在(反覆出現)的各種問題,所提出的解決方案。 ...
  • 面試官:請問HashSet有哪些特點? 應聘者:HashSet實現自set介面,set集合中元素無序且不能重覆; 面試官:那麼HashSet 如何保證元素不重覆? 應聘者:因為HashSet底層是基於HashMap實現的,當你new一個HashSet時候,實際上是new了一個map,執行add方法時 ...
  • 解析調用   方法調用的目標方法在Class文件里是一個常量池中的符號引用,在類載入的解析階段,將其中一部分符號引用轉化為直接引用,這種解析的前提是:方法在程式真正運行之前就有一個可確定的調用版本,並且這個方法的調用版本在運行期不可變(編譯期可知,運行器不可變)。這類方法的調用稱 ...
  • --基於譚浩強老師《C++程式設計(第三版)》做簡要Summary。(2019-07-24) 一、數組與指針 1. 指針數組(type_name * array_name[length]) 一個數組,其元素均為指針類型數據,該數組稱為指針數組。 2. 數組指針 二維數組的指針訪問: 二、 const ...
  • 其實在JAVA開發中servlet配置,映射註入配置等等都可以用xml來配置 在此處的department是實體類的名字,而不是對應的資料庫表的名字 資料庫表的欄位名=#{實體類屬性名} 逆向工程生成的XML文件有查找更新等功能,但是當我們查找的時候需要返回一個類, 我們應該在開頭寫返回結果 res ...
  • 在Java開發中,我們最常見到最頻繁使用的就是HashMap和HashTable,但是線上程競爭激烈的併發場景中使用都不夠合理。 1、HashMap 眾所周知 HashMap 底層是基於數組 + 鏈表組成的,不過在 jdk1.7 與1.8 中具體實現稍有不同。 HashMap是線程不安全的,在併發( ...
  • capture的作用是: 捕獲模板輸出的數據並將其存儲到一個變數,而不是把它們輸出到頁面,任何在 {capture name="foo"}和{/capture}之間的數據將被存儲到變數$foo中,該變數由name屬性指定,在模板中通過 $smarty.capture.foo 訪問該變數,{captu ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...