1. 程式人生 > 實用技巧 >(2)函式之物件、巢狀、名稱空間與作用域、閉包函式、裝飾器

(2)函式之物件、巢狀、名稱空間與作用域、閉包函式、裝飾器

一、函式物件


1,函式是第一類物件,即函式可以當做資料傳遞

# 1,可以被引用
# 2,可以當作引數傳遞
# 3,返回值可以是函式
# 3,可以當作容器型別的元素

2,利用該特性,優雅的取代多分支的 if

def foo():
    print('foo')

def bar():
    print('bar')

dic={
    'foo':foo,
    'bar':bar,
}

while True:
    choice = input('>>: ').strip()
    if choice in dic:
        dic[choice]()

二、函式巢狀


1,函式的巢狀呼叫

def max(x,y):
    return x if x > y else y

def max4(a,b,c,d):
    res1 = max(a,b)
    res2 = max(res1,c)
    res3 = max(res2,d)
    return res3
print(max4(1,2,3,4))

2,函式的巢狀定義

def f1():
    def f2():
        def f3():
            print('from f3')
        f3()
    f2()

f1()
f3()     
# 報錯,為何?看下一節

三、名稱空間與作用域


1,什麼是名稱空間?

名稱空間是存放名字的地方(之前遺留的問題 x = 1,1存放於記憶體中,那名字 x 存放在哪裡了?名稱空間正是存放名字 x 與 1 繫結關係的地方)。

2,名稱空間的載入順序

python test.py
# 1,python直譯器先啟動,因而首先載入的是:內建名稱空間
# 2,執行test.py檔案,然後以檔案為基礎,載入全域性名稱空間
# 3,在執行檔案的過程中如果呼叫函式,則臨時產生區域性名稱空間

3,名字的查詢順序

區域性名稱空間--->全域性名稱空間--->內建名稱空間

# 需要注意的是:在全域性無法檢視區域性的,在區域性可以檢視全域性的,如下示例:
# max=1 def f1(): # max=2 def f2(): # max=3 print(max) f2() f1() print(max)

4,作用域

# 1,作用域即範圍
        - 全域性範圍(內建名稱空間與全域性名稱空間屬於該範圍):全域性存活,全域性有效
      - 區域性範圍(區域性名稱空間屬於該範圍):臨時存活,區域性有效


# 2,作用域關係是在函式定義階段就已經固定的,與函式的呼叫位置無關,如下:
x=1
def f1():
    def f2():
        print(x)
    return f2
x=100
def f3(func):
    x=2
    func()
x=10000
f3(f1())


# 3,檢視作用域:globals(),locals()
LEGB 代表名字查詢順序: locals -> enclosing function -> globals -> __builtins__
locals 是函式內的名字空間,包括區域性變數和形參
enclosing 外部巢狀函式的名字空間(閉包中常見)
globals 全域性變數,函式定義所在模組的名字空間
builtins 內建模組的名字空間

四、閉包函式


1,什麼是閉包?

# 內部函式包含對外部作用域而非全域性作用域的引用

# 提示:之前我們都是通過引數將外部的值傳給函式,閉包提供了另外一種思路,包起來嘍,包起呦,包起來哇。

        def counter():
            n=0
            def incr():
                nonlocal n
                x=n
                n+=1
                return x
            return incr

        c=counter()
        print(c())
        print(c())
        print(c())
        print(c.__closure__[0].cell_contents)     # 檢視閉包的元素

2,閉包的意義與應用

# 閉包的意義:返回的函式物件,不僅僅是一個函式物件,在該函式外還包裹了一層作用域,這使得,該函式無論在何處呼叫,優先使用自己外層包裹的作用域。

# 應用領域:延遲計算(原來我們是傳參,現在我們是包起來)
    from urllib.request import urlopen

    def index(url):
        def get():
            return urlopen(url).read()
        return get

    baidu = index('http://www.baidu.com')
    print(baidu().decode('utf-8'))

五、裝飾器


裝飾器就是閉包函式的一種應用場景。

1,為什麼要用裝飾器?

因為開放封閉原則:對修改封閉,對擴充套件開放

2,什麼是裝飾器?

# 裝飾他人的器具,本身可以是任意可呼叫物件,被裝飾者也可以是任意可呼叫物件。

# 強調裝飾器的原則:
# 1)不修改被裝飾物件的原始碼。
# 2)不修改被裝飾物件的呼叫方式。

# 裝飾器的目標:在遵循1)和2)的前提下,為被裝飾物件新增上新功能

3,裝飾器的使用

import time

def timmer(func):
    def wrapper(*args,**kwargs):
        start_time = time.time()
        res = func(*args,**kwargs)
        stop_time = time.time()
        print('run time is %s' % (stop_time-start_time))
        return res
    return wrapper

@timmer
def foo():
    time.sleep(3)
    print('from foo')
foo()

# from foo
# run time is 3.0001039505004883
無參裝飾器
def auth(driver='file'):
    def auth2(func):
        def wrapper(*args,**kwargs):
            name = input("user: ")
            pwd = input("pwd: ")

            if driver == 'file':
                if name == 'zixi' and pwd == '123':
                    print('login successful')
                    res = func(*args,**kwargs)
                    return res
            elif driver == 'ldap':
                print('ldap')
        return wrapper
    return auth2

@auth(driver='file')
def foo(name):
    print(name)

foo('zixi')

"""
user: zixi
pwd: 123
login successful
zixi
"""
有參裝飾器

4,裝飾器語法

被裝飾函式的正上方,單獨一行

        @deco1
        @deco2
        @deco3
        def foo():
            pass

        foo = deco1(deco2(deco3(foo)))

5,裝飾器補充:wraps

from functools import wraps

def deco(func):
    @wraps(func)    # 加在最內層函式正上方
    def wrapper(*args,**kwargs):
        return func(*args,**kwargs)
    return wrapper

@deco
def index():
    """哈哈哈"""
    print('from index')

print(index.__doc__)    # 列印註釋

6,疊加多個裝飾器

# 疊加多個裝飾器
# 1,載入順序(outter函式的呼叫順序):自下而上
# 2,執行順序(wrapper函式的執行順序):自上而下
def outter1(func1):     # func1=wrapper2的記憶體地址
    print('載入了outter1')
    def wrapper1(*args,**kwargs):
        print('執行了wrapper1')
        res1 = func1(*args,**kwargs)
        return res1
    return wrapper1

def outter2(func2):     # func2=wrapper3的記憶體地址
    print('載入了outter2')
    def wrapper2(*args,**kwargs):
        print('執行了wrapper2')
        res2 = func2(*args,**kwargs)
        return res2
    return wrapper2

def outter3(func3):     # func3=最原始的那個index的記憶體地址
    print('載入了outter3')
    def wrapper3(*args,**kwargs):
        print('執行了wrapper3')
        res3 = func3(*args,**kwargs)
        return res3
    return wrapper3



@outter1    # outter1(wrapper2的記憶體地址)======>index=wrapper1的記憶體地址
@outter2    # outter2(wrapper3的記憶體地址)======>wrapper2的記憶體地址
@outter3    # outter3(最原始的那個index的記憶體地址)===>wrapper3的記憶體地址
def index():
    print('from index')

print('======================================================')
index()

六、練習


1,編寫函式,(函式執行的時間是隨機的)

2,編寫裝飾器,為函式加上統計時間的功能

3,編寫裝飾器,為函式加上認證的功能

4,編寫裝飾器,為多個函式加上認證的功能(使用者的賬號密碼來源於檔案),要求登入成功一次,後續的函式都無需再輸入使用者名稱和密碼。
注意:從檔案中讀出字串形式的字典,可以用eval('{"name":"egon","password":"123"}')轉成字典格式。

5,編寫裝飾器,為多個函式加上認證功能,要求登入成功一次,在超時時間內無需重複登入,超過了超時時間,則必須重新登入。

6,編寫下載網頁內容的函式,要求功能是:使用者傳入一個url,函式返回下載頁面的結果

7,為題目五編寫裝飾器,實現快取網頁內容的功能:
具體:實現下載的頁面存放於檔案中,如果檔案內有值(檔案大小不為0),就優先從檔案中讀取網頁內容,否則,就去下載,然後存到檔案中。

擴充套件功能:使用者可以選擇快取介質/快取引擎,針對不同的url,快取到不同的檔案中

8,還記得我們用函式物件的概念,製作一個函式字典的操作嗎,來來來,我們有更高大上的做法,在檔案開頭宣告一個空字典,然後在每個函式前加上裝飾器,完成自動新增到字典的操作

9,編寫日誌裝飾器,實現功能如:一旦函式f1執行,則將訊息2017-07-21 11:12:11 f1 run寫入到日誌檔案中,日誌檔案路徑可以指定。
注意:時間格式的獲取
import time
time.strftime('%Y-%m-%d %X')

#題目一:略
#題目二:略
#題目三:略
#題目四:
db='db.txt'
login_status={'user':None,'status':False}
def auth(auth_type='file'):
    def auth2(func):
        def wrapper(*args,**kwargs):
            if login_status['user'] and login_status['status']:
                return func(*args,**kwargs)
            if auth_type == 'file':
                with open(db,encoding='utf-8') as f:
                    dic=eval(f.read())
                name=input('username: ').strip()
                password=input('password: ').strip()
                if name in dic and password == dic[name]:
                    login_status['user']=name
                    login_status['status']=True
                    res=func(*args,**kwargs)
                    return res
                else:
                    print('username or password error')
            elif auth_type == 'sql':
                pass
            else:
                pass
        return wrapper
    return auth2

@auth()
def index():
    print('index')

@auth(auth_type='file')
def home(name):
    print('welcome %s to home' %name)


# index()
# home('egon')

#題目五
import time,random
user={'user':None,'login_time':None,'timeout':0.000003,}

def timmer(func):
    def wrapper(*args,**kwargs):
        s1=time.time()
        res=func(*args,**kwargs)
        s2=time.time()
        print('%s' %(s2-s1))
        return res
    return wrapper


def auth(func):
    def wrapper(*args,**kwargs):
        if user['user']:
            timeout=time.time()-user['login_time']
            if timeout < user['timeout']:
                return func(*args,**kwargs)
        name=input('name>>: ').strip()
        password=input('password>>: ').strip()
        if name == 'egon' and password == '123':
            user['user']=name
            user['login_time']=time.time()
            res=func(*args,**kwargs)
            return res
    return wrapper

@auth
def index():
    time.sleep(random.randrange(3))
    print('welcome to index')

@auth
def home(name):
    time.sleep(random.randrange(3))
    print('welcome %s to home ' %name)

index()
home('egon')

#題目六:略
#題目七:簡單版本
import requests
import os
cache_file='cache.txt'
def make_cache(func):
    def wrapper(*args,**kwargs):
        if not os.path.exists(cache_file):
            with open(cache_file,'w'):pass

        if os.path.getsize(cache_file):
            with open(cache_file,'r',encoding='utf-8') as f:
                res=f.read()
        else:
            res=func(*args,**kwargs)
            with open(cache_file,'w',encoding='utf-8') as f:
                f.write(res)
        return res
    return wrapper

@make_cache
def get(url):
    return requests.get(url).text


# res=get('https://www.python.org')

# print(res)

#題目七:擴充套件版本
import requests,os,hashlib
engine_settings={
    'file':{'dirname':'./db'},
    'mysql':{
        'host':'127.0.0.1',
        'port':3306,
        'user':'root',
        'password':'123'},
    'redis':{
        'host':'127.0.0.1',
        'port':6379,
        'user':'root',
        'password':'123'},
}

def make_cache(engine='file'):
    if engine not in engine_settings:
        raise TypeError('egine not valid')
    def deco(func):
        def wrapper(url):
            if engine == 'file':
                m=hashlib.md5(url.encode('utf-8'))
                cache_filename=m.hexdigest()
                cache_filepath=r'%s/%s' %(engine_settings['file']['dirname'],cache_filename)

                if os.path.exists(cache_filepath) and os.path.getsize(cache_filepath):
                    return open(cache_filepath,encoding='utf-8').read()

                res=func(url)
                with open(cache_filepath,'w',encoding='utf-8') as f:
                    f.write(res)
                return res
            elif engine == 'mysql':
                pass
            elif engine == 'redis':
                pass
            else:
                pass

        return wrapper
    return deco

@make_cache(engine='file')
def get(url):
    return requests.get(url).text

# print(get('https://www.python.org'))
print(get('https://www.baidu.com'))


#題目八
route_dic={}

def make_route(name):
    def deco(func):
        route_dic[name]=func
    return deco
@make_route('select')
def func1():
    print('select')

@make_route('insert')
def func2():
    print('insert')

@make_route('update')
def func3():
    print('update')

@make_route('delete')
def func4():
    print('delete')

print(route_dic)


#題目九
import time
import os

def logger(logfile):
    def deco(func):
        if not os.path.exists(logfile):
            with open(logfile,'w'):pass

        def wrapper(*args,**kwargs):
            res=func(*args,**kwargs)
            with open(logfile,'a',encoding='utf-8') as f:
                f.write('%s %s run\n' %(time.strftime('%Y-%m-%d %X'),func.__name__))
            return res
        return wrapper
    return deco

@logger(logfile='aaaaaaaaaaaaaaaaaaaaa.log')
def index():
    print('index')

index()
result