利用Python的裝飾器解決Bottle框架中用戶驗證問題
首先來分析下需求,web程序后臺需要認(rèn)證,后臺頁面包含多個頁面,最普通的方法就是為每個url添加認(rèn)證,但是這樣就需要每個每個綁定url的后臺函數(shù)都需要添加類似或者相同的代碼,但是這樣做代碼就過度冗余,而且不利于擴(kuò)展.
接下來我們先不談及裝飾器,我們都知道Python是個很強(qiáng)大的語言,她可以將函數(shù)當(dāng)做參數(shù)傳遞給函數(shù),最簡單的:
def p(): print 'Hello,world' def funcfactor(func): print 'calling function named', func.__name__ func() print 'end' funcfactor(p) # 輸出為: # calling function named p # Hello,world # end
一目了然的程序,定義一個函數(shù)p(),將函數(shù)p當(dāng)做參數(shù)傳遞給喊出funcfactor,在執(zhí)行p函數(shù)前后加上一些動作.
我們還可以這么做:
def p(): print 'Hello,world' def funcfactor(func): print 'calling function named', func.__name__ return func func = funcfactor(p) func() # 輸出為: # calling function named p Hello,world
正如你看到的,我們可以將函數(shù)返回然后賦予一個變量,留待稍后調(diào)用.但是這種情況下我們要想在函數(shù)執(zhí)行后做點什么就不可能,但是我們的Python是強(qiáng)大的,Python可以在函數(shù)中再嵌套一個函數(shù),我們可以像下面這么做:
def p(): print 'Hello, world' def funcfactor(func): def wrapper(): print 'do something at start' func() print 'do something at end' return wrapper func = funcfactor(p) func() #輸出為: # do something at start # Hello, world # do something at end
下面我們來看看裝飾器,上面的代碼雖然實現(xiàn)的一個很困難的任務(wù),但是還不夠優(yōu)雅,而且代碼不符合Python的哲學(xué)思想,所以裝飾器就應(yīng)聲而出,裝飾器沒有和上面的原理相同,同樣用于包裝函數(shù),只是代碼實現(xiàn)上更加優(yōu)雅和便于閱讀.裝飾器以@開頭后面跟上裝飾器的名稱,緊接著下一行就是要包裝的函數(shù)體,上面的例子用裝飾器可用如下方式實現(xiàn):
def decorator(func): def wrapper(): print 'do something at start' func() print 'do something at end' return wrapper @decorator def p(): print 'Hello, world' p() #輸出為: # do something at start # Hello, world # do something at end
實際上裝飾器并沒有性能方面或其他方面的提升,僅僅是一種語法糖,就是上面一個例子的改寫,這樣更加優(yōu)雅和便與閱讀. 如果我們的p()函數(shù)不想僅僅只輸Hello,world,我們想向某些我們指定的人打招呼:
def decorator(func): def wrapper(*args, **kargs): print 'do something at start' func(**kargs) print 'do something at end' return wrapper @decorator def p(name): print 'Hello', name p(name="Jim") #輸出為: # do something at start # Hello Jim # do something at end
裝飾器在裝飾不需要參數(shù)的裝飾器嵌套函數(shù)不是必須得,如果被裝飾的函數(shù)需要參數(shù),必須嵌套一個函數(shù)來處理參數(shù). 寫到這里想必大家也知道裝飾器的用法和作用.現(xiàn)在回到正題,如何優(yōu)雅的給后臺url加上驗證功能?毫無疑問我們使用裝飾器來處理:
def blog_auth(func): ''' 定義一個裝飾器用于裝飾需要驗證的頁面 裝飾器必須放在route裝飾器下面 ''' # 定義包裝函數(shù) def wrapper(*args, **kargs): try: # 讀取cookie user = request.COOKIES['user'] shell = request.COOKIES['shell'] except: # 出現(xiàn)異常則重定向到登錄頁面 redirect('/login') # 驗證用戶數(shù)據(jù) if checkShell(user, shell): # 校驗成功則返回函數(shù) return func(**kargs) else: # 否則則重定向到登錄頁面 redirect('/login') return wrapper
可以再需要驗證的地方添加blog_auth裝飾器:
@route('/admin:#/?#') @blog_auth def admin(): ''' 用于顯示后臺管理首頁 ''' TEMPLATE['title'] = '儀表盤 | ' + TEMPLATE['BLOG_NAME'] TEMPLATE['user'] = request.COOKIES['user'] articles = [] for article in db.posts.find().sort("date",DESCENDING).limit(10): articles.append(article) # 將文章列表交給前臺模版 TEMPLATE['articles'] = articles return template('admin.html',TEMPLATE)
至此bottle驗證的問題就很優(yōu)雅的用裝飾器解決了.
相關(guān)文章
用Python批量把文件復(fù)制到另一個文件夾的實現(xiàn)方法
這篇文章主要介紹了用Python批量把文件復(fù)制到另一個文件夾的實現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08pybaobabdt庫基于python的決策樹隨機(jī)森林可視化工具使用
這篇文章主要為大家介紹了pybaobabdt庫基于python的決策樹隨機(jī)森林可視化工具使用探索,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-02-02Django上使用數(shù)據(jù)可視化利器Bokeh解析
這篇文章主要介紹了Django上使用數(shù)據(jù)可視化利器Bokeh解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2019-07-07PyCharm調(diào)用matplotlib繪圖時圖像彈出問題詳解
這篇文章主要給大家介紹了關(guān)于PyCharm調(diào)用matplotlib繪圖時圖像彈出問題的相關(guān)資料,文中通過圖文介紹的非常詳細(xì),對大家學(xué)習(xí)或者使用PyCharm具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2022-07-07Python虛擬環(huán)境virtualenv的安裝與使用詳解
virtualenv可以用來管理互不干擾的獨立python虛擬環(huán)境,在有些場景下非常有用,下面這篇文章主要給大家介紹了Python虛擬環(huán)境virtualenv安裝與使用的相關(guān)資料,需要的朋友可以參考借鑒,下面來一起看看吧。2017-05-05python實現(xiàn)樹的深度優(yōu)先遍歷與廣度優(yōu)先遍歷詳解
這篇文章主要介紹了python實現(xiàn)樹的深度優(yōu)先遍歷與廣度優(yōu)先遍歷,詳細(xì)分析了樹的深度優(yōu)先遍歷與廣度優(yōu)先遍歷原理及Python相關(guān)實現(xiàn)技巧,需要的朋友可以參考下2019-10-10使用pip下載時提示"You?are?using?pip?version?8.1.1,?howev
最近在使用python的pip下載庫時,出現(xiàn)了報錯,所以下面這篇文章主要給大家介紹了關(guān)于使用pip下載時提示“You?are?using?pip?version?8.1.1,?however?version?22.1?is?available.“錯誤的解決方法,需要的朋友可以參考下2022-08-08