第九篇 裝飾器

閱讀目錄

一 什麼是裝飾器

二 裝飾器需要遵循的原則

三 實現裝飾器知識儲備

四 高階函數

五 函數嵌套

六 閉包

七 無參裝飾器

八 裝飾器應用示例

 

一 什麼是裝飾器


器即函數


裝飾即修飾,意指爲其他函數添加新功能


裝飾器定義:本質就是函數,功能是爲其他函數添加新功能


 

二 裝飾器需要遵循的原則


1.不修改被裝飾函數的源代碼(開放封閉原則)


2.爲被裝飾函數添加新功能後,不修改被修飾函數的調用方式


 

三 實現裝飾器知識儲備


裝飾器=高階函數+函數嵌套+閉包


 

四 高階函數


高階函數定義:

1.函數接收的參數是一個函數名


2.函數的返回值是一個函數名


3.滿足上述條件任意一個,都可稱之爲高階函數



 def foo():

    print('我的函數名作爲參數傳給高階函數')

def gao_jie1(func):

    print('我就是高階函數1,我接收的參數名是%s' %func)

    func()


def gao_jie2(func):

    print('我就是高階函數2,我的返回值是%s' %func)

    return func


gao_jie1(foo)

gao_jie2(foo)

 

 #高階函數應用1:把函數當做參數傳給高階函數

import time

def foo():

    print('from the foo')


def timmer(func):

    start_time=time.time()

    func()

    stop_time=time.time()

    print('函數%s 運行時間是%s' %(func,stop_time-start_time))

timmer(foo)

#總結:我們確實爲函數foo增加了foo運行時間的功能,但是foo原來的執行方式是foo(),現在我們需要調用高階函數timmer(foo),改變了函數的調用方式

 

 #高階函數應用2:把函數名當做參數傳給高階函數,高階函數直接返回函數名

import time

def foo():

    print('from the foo')


def timmer(func):

    start_time=time.time()

    return func

    stop_time=time.time()

    print('函數%s 運行時間是%s' %(func,stop_time-start_time))

foo=timmer(foo)

foo()

#總結:我們確實沒有改變foo的調用方式,但是我們也沒有爲foo增加任何新功能

 高階函數總結

1.函數接收的參數是一個函數名

  作用:在不修改函數源代碼的前提下,爲函數添加新功能,

  不足:會改變函數的調用方式

2.函數的返回值是一個函數名

  作用:不修改函數的調用方式

  不足:不能添加新功能


 

五 函數嵌套


  1 def father(name):

 2     print('from father %s' %name)

 3     def son():

 4         print('from son')

 5         def grandson():

 6             print('from grandson')

 7         grandson()

 8     son()

 9 

10 father('tom')

  

六 閉包


  1 '''

 2 閉包:在一個作用域裏放入定義變量,相當於打了一個包

 3 '''

 4 def father(name):

 5     def son():

 6         # name='jack'

 7         print('is [%s]' %name)

 8         def grandson():

 9             # name='wupeiqi'

10             print('is [%s]' %name)

11         grandson()

12     son()

13 

14 father('jack')

  

七 無參裝飾器


無參裝飾器=高級函數+函數嵌套


基本框架


1 #這就是一個實現一個裝飾器最基本的架子

2 def timer(func):

3     def wrapper():

4         func()

5     return wrapper

加上參數


1 def timer(func):

2     def wrapper(*args,**kwargs):

3         func(*args,**kwargs)

4     return wrapper

加上功能


 1 import time

2 def timer(func):

3     def wrapper(*args,**kwargs):

4         start_time=time.time()

5         func(*args,**kwargs)

6         stop_time=time.time()

7         print('函數[%s],運行時間是[%s]' %(func,stop_time-start_time))

8     return wrapper

 加上返回值


 1 import time

2 def timer(func):

3     def wrapper(*args,**kwargs):

4         start_time=time.time()

5         res=func(*args,**kwargs)

6         stop_time=time.time()

7         print('函數[%s],運行時間是[%s]' %(func,stop_time-start_time))

8         return res

9     return wrapper

 使用裝飾器


 1 def cal(array):

2     res=0

3     for i in array:

4         res+=i

5     return res

7 cal=timer(cal)

8 cal(range(10))

 語法糖@


 1 @timer  #@timer就等同於cal=timer(cal)

2 def cal(array):

3     res=0

4     for i in array:

5         res+=i

6     return res

8 cal(range(10))

  

八 裝飾器應用示例



 user_list=[

    {'name':'alex','passwd':'123'},

    {'name':'linhaifeng','passwd':'123'},

    {'name':'wupeiqi','passwd':'123'},

    {'name':'yuanhao','passwd':'123'},

]


current_user={'username':None,'login':False}


def auth_deco(func):

    def wrapper(*args,**kwargs):

        if current_user['username'] and current_user['login']:

            res=func(*args,**kwargs)

            return res

        username=input('用戶名: ').strip()

        passwd=input('密碼: ').strip()


        for index,user_dic in enumerate(user_list):

            if username == user_dic['name'] and passwd == user_dic['passwd']:

                current_user['username']=username


                current_user['login']=True

                res=func(*args,**kwargs)

                return res

                break

        else:

            print('用戶名或者密碼錯誤,重新登錄')


    return wrapper


@auth_deco

def index():

    print('歡迎來到主頁面')


@auth_deco

def home():

    print('這裏是你家')


def shopping_car():

    print('查看購物車啊親')


def order():

    print('查看訂單啊親')


print(user_list)

# index()

print(user_list)

home()

 

 user_list=[

    {'name':'tom','passwd':'123'},

    {'name':'jack','passwd':'123'},

  ]


current_user={'username':None,'login':False}

def auth(auth_type='file'):

    def auth_deco(func):

        def wrapper(*args,**kwargs):

            if auth_type == 'file':

                if current_user['username'] and current_user['login']:

                    res=func(*args,**kwargs)

                    return res

                username=input('用戶名: ').strip()

                passwd=input('密碼: ').strip()


                for index,user_dic in enumerate(user_list):

                    if username == user_dic['name'] and passwd == user_dic['passwd']:

                        current_user['username']=username

                        current_user['login']=True

                        res=func(*args,**kwargs)

                        return res

                        break

                else:

                    print('用戶名或者密碼錯誤,重新登錄')

            elif auth_type == 'ldap':

                print('auth')

                res=func(*args,**kwargs)

                return res

        return wrapper

    return auth_deco



#auth(auth_type='file')就是在運行一個函數,然後返回auth_deco,所以@auth(auth_type='file')

#就相當於@auth_deco,只不過現在,我們的auth_deco作爲一個閉包的應用,外層的包auth給它留了一個auth_type='file'參數

@auth(auth_type='ldap')

def index():

    print('歡迎來到主頁面')


@auth(auth_type='ldap')

def home():

    print('home')


def shopping_car():

    print('shopping')


def order():

    print('check order')


# print(user_list)

index()

# print(user_list)

home()


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章