欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Python裝飾器用法實(shí)例總結(jié)

 更新時(shí)間:2018年02月07日 10:30:44   作者:蒼松  
這篇文章主要介紹了Python裝飾器用法,結(jié)合實(shí)例形式總結(jié)分析了Python常用裝飾器的概念、功能、使用方法及相關(guān)注意事項(xiàng),需要的朋友可以參考下

本文實(shí)例講述了Python裝飾器用法。分享給大家供大家參考,具體如下:

一、裝飾器是什么

python的裝飾器本質(zhì)上是一個(gè)Python函數(shù),它可以讓其他函數(shù)在不需要做任何代碼變動(dòng)的前提下增加額外功能,裝飾器的返回值也是一個(gè)函數(shù)對(duì)象。簡(jiǎn)單的說裝飾器就是一個(gè)用來返回函數(shù)的函數(shù)。

它經(jīng)常用于有切面需求的場(chǎng)景,比如:插入日志、性能測(cè)試、事務(wù)處理、緩存、權(quán)限校驗(yàn)等場(chǎng)景。裝飾器是解決這類問題的絕佳設(shè)計(jì),有了裝飾器,我們就可以抽離出大量與函數(shù)功能本身無關(guān)的雷同代碼并繼續(xù)重用。

概括的講,裝飾器的作用就是為已經(jīng)存在的對(duì)象添加額外的功能。

二、為什么需要裝飾器

1、先來看一個(gè)簡(jiǎn)單例子:

def foo():
print('i am foo')

2、增加需求

現(xiàn)在有一個(gè)新的需求,希望可以記錄下函數(shù)的執(zhí)行日志,于是在代碼中添加日志代碼:

def foo():
  print('i am foo')
  print("foo is running")

3、又有需求

假設(shè)現(xiàn)在有100個(gè)函數(shù)需要增加這個(gè)需求,并且后續(xù)可能還要對(duì)這一百個(gè)函數(shù)都增加執(zhí)行前打印日志的需求,怎么辦?還一個(gè)個(gè)改嗎?

當(dāng)然不了,這樣會(huì)造成大量雷同的代碼,為了減少重復(fù)寫代碼,我們可以這樣做,重新定義一個(gè)函數(shù):專門處理日志 ,日志處理完之后再執(zhí)行真正的業(yè)務(wù)代碼。

def use_logging(func):
  print("%s is running" % func.__name__)
  func()
def bar():
  print('i am bar')
use_logging(bar)
#result:
#bar is running
#i am bar

通過以上use_logging函數(shù)我們?cè)黾恿巳罩竟δ?,不管以后有多少函?shù)需要增加日志或者修改日志的格式我們只需要修改use_logging函數(shù),并執(zhí)行use_logging(被裝飾的函數(shù))就達(dá)到了我們想要的效果。

def use_logging(func):
  print("%s is running" % func.__name__)
  return func
@use_logging
def bar():
  print('i am bar')
bar()

三、基礎(chǔ)裝飾器入門

1、裝飾器語法糖

python提供了@符號(hào)作為裝飾器的語法糖,使我們更方便的應(yīng)用裝飾函數(shù)。但使用語法糖要求裝飾函數(shù)必須return一個(gè)函數(shù)對(duì)象。因此我們將上面的func函數(shù)使用內(nèi)嵌函數(shù)包裹并return。

裝飾器相當(dāng)于執(zhí)行了裝飾函數(shù)use_loggin后又返回被裝飾函數(shù)bar,因此bar()被調(diào)用的時(shí)候相當(dāng)于執(zhí)行了兩個(gè)函數(shù)。等價(jià)于use_logging(bar)()

def use_logging(func):
  def _deco():
    print("%s is running" % func.__name__)
    func()
  return _deco
@use_logging
def bar():
  print('i am bar')
bar()

2、對(duì)帶參數(shù)的函數(shù)進(jìn)行裝飾

現(xiàn)在我們的參數(shù)需要傳入兩個(gè)參數(shù)并計(jì)算值,因此我們需要對(duì)內(nèi)層函數(shù)進(jìn)行改動(dòng)傳入我們的兩個(gè)參數(shù)a和b,等價(jià)于use_logging(bar)(1,2)

def use_logging(func):
  def _deco(a,b):
    print("%s is running" % func.__name__)
    func(a,b)
  return _deco
@use_logging
def bar(a,b):
  print('i am bar:%s'%(a+b))
bar(1,2)

我們裝飾的函數(shù)可能參數(shù)的個(gè)數(shù)和類型都不一樣,每一次我們都需要對(duì)裝飾器做修改嗎?這樣做當(dāng)然是不科學(xué)的,因此我們使用python的變長(zhǎng)參數(shù)*args和**kwargs來解決我們的參數(shù)問題。

3、函數(shù)參數(shù)數(shù)量不確定

不帶參數(shù)裝飾器版本,這個(gè)格式適用于不帶參數(shù)的裝飾器。

經(jīng)過以下修改我們已經(jīng)適應(yīng)了各種長(zhǎng)度和類型的參數(shù)。這個(gè)版本的裝飾器已經(jīng)可以任意類型的無參數(shù)函數(shù)。

def use_logging(func):
  def _deco(*args,**kwargs):
    print("%s is running" % func.__name__)
    func(*args,**kwargs)
  return _deco
@use_logging
def bar(a,b):
  print('i am bar:%s'%(a+b))
@use_logging
def foo(a,b,c):
  print('i am bar:%s'%(a+b+c))
bar(1,2)
foo(1,2,3)

4、裝飾器帶參數(shù)

帶參數(shù)的裝飾器,這個(gè)格式適用于帶參數(shù)的裝飾器。

某些情況我們需要讓裝飾器帶上參數(shù),那就需要編寫一個(gè)返回一個(gè)裝飾器的高階函數(shù),寫出來會(huì)更復(fù)雜。比如:

#! /usr/bin/env python
# -*- coding:utf-8 -*-
# __author__ = "TKQ"
def use_logging(level):
  def _deco(func):
    def __deco(*args, **kwargs):
      if level == "warn":
        print "%s is running" % func.__name__
      return func(*args, **kwargs)
    return __deco
  return _deco
@use_logging(level="warn")
def bar(a,b):
  print('i am bar:%s'%(a+b))
bar(1,3)
# 等價(jià)于use_logging(level="warn")(bar)(1,3)

5、functools.wraps

使用裝飾器極大地復(fù)用了代碼,但是他有一個(gè)缺點(diǎn)就是原函數(shù)的元信息不見了,比如函數(shù)的docstring、__name__、參數(shù)列表,先看例子:

def use_logging(func):
  def _deco(*args,**kwargs):
    print("%s is running" % func.__name__)
    func(*args,**kwargs)
  return _deco
@use_logging
def bar():
  print('i am bar')
  print(bar.__name__)
bar()
#bar is running
#i am bar
#_deco
#函數(shù)名變?yōu)開deco而不是bar,這個(gè)情況在使用反射的特性的時(shí)候就會(huì)造成問題。因此引入了functools.wraps解決這個(gè)問題。

使用functools.wraps:

import functools
def use_logging(func):
  @functools.wraps(func)
  def _deco(*args,**kwargs):
    print("%s is running" % func.__name__)
    func(*args,**kwargs)
  return _deco
@use_logging
def bar():
  print('i am bar')
  print(bar.__name__)
bar()
#result:
#bar is running
#i am bar
#bar ,這個(gè)結(jié)果是我們想要的。OK啦!

6、實(shí)現(xiàn)帶參數(shù)和不帶參數(shù)的裝飾器自適應(yīng)

import functools
def use_logging(arg):
  if callable(arg):#判斷參入的參數(shù)是否是函數(shù),不帶參數(shù)的裝飾器調(diào)用這個(gè)分支
    @functools.wraps(arg)
    def _deco(*args,**kwargs):
      print("%s is running" % arg.__name__)
      arg(*args,**kwargs)
    return _deco
  else:#帶參數(shù)的裝飾器調(diào)用這個(gè)分支
    def _deco(func):
      @functools.wraps(func)
      def __deco(*args, **kwargs):
        if arg == "warn":
          print "warn%s is running" % func.__name__
        return func(*args, **kwargs)
      return __deco
    return _deco
@use_logging("warn")
# @use_logging
def bar():
  print('i am bar')
  print(bar.__name__)
bar()

三、類裝飾器

使用類裝飾器可以實(shí)現(xiàn)帶參數(shù)裝飾器的效果,但實(shí)現(xiàn)的更加優(yōu)雅簡(jiǎn)潔,而且可以通過繼承來靈活的擴(kuò)展.

1、類裝飾器

class loging(object):
  def __init__(self,level="warn"):
    self.level = level
  def __call__(self,func):
    @functools.wraps(func)
    def _deco(*args, **kwargs):
      if self.level == "warn":
        self.notify(func)
      return func(*args, **kwargs)
    return _deco
  def notify(self,func):
    # logit只打日志,不做別的
    print "%s is running" % func.__name__
@loging(level="warn")#執(zhí)行__call__方法
def bar(a,b):
  print('i am bar:%s'%(a+b))
bar(1,3)

2、繼承擴(kuò)展類裝飾器

class email_loging(Loging):
  '''
  一個(gè)loging的實(shí)現(xiàn)版本,可以在函數(shù)調(diào)用時(shí)發(fā)送email給管理員
  '''
  def __init__(self, email='admin@myproject.com', *args, **kwargs):
    self.email = email
    super(email_loging, self).__init__(*args, **kwargs)
  def notify(self,func):
    # 發(fā)送一封email到self.email
    print "%s is running" % func.__name__
    print "sending email to %s" %self.email
@email_loging(level="warn")
def bar(a,b):
  print('i am bar:%s'%(a+b))
bar(1,3)

更多關(guān)于Python相關(guān)內(nèi)容可查看本站專題:《Python數(shù)據(jù)結(jié)構(gòu)與算法教程》、《Python Socket編程技巧總結(jié)》、《Python函數(shù)使用技巧總結(jié)》、《Python字符串操作技巧匯總》及《Python入門與進(jìn)階經(jīng)典教程

希望本文所述對(duì)大家Python程序設(shè)計(jì)有所幫助。

相關(guān)文章

  • python音頻處理的示例詳解

    python音頻處理的示例詳解

    這篇文章主要介紹了python音頻處理的示例詳解,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-12-12
  • Python Selenium異常處理的實(shí)例分析

    Python Selenium異常處理的實(shí)例分析

    在本篇內(nèi)容里小編給大家分享了關(guān)于Python Selenium異常處理的實(shí)例分析內(nèi)容,對(duì)此有興趣的朋友們可以學(xué)習(xí)參考下。
    2021-02-02
  • 如何解決vscode下powershell終端進(jìn)入python虛擬環(huán)境venv問題

    如何解決vscode下powershell終端進(jìn)入python虛擬環(huán)境venv問題

    這篇文章主要介紹了如何解決vscode下powershell終端進(jìn)入python虛擬環(huán)境venv問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-05-05
  • 淺談sklearn中predict與predict_proba區(qū)別

    淺談sklearn中predict與predict_proba區(qū)別

    這篇文章主要介紹了淺談sklearn中predict與predict_proba區(qū)別,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2020-06-06
  • Pycharm學(xué)習(xí)教程(3) 代碼運(yùn)行調(diào)試

    Pycharm學(xué)習(xí)教程(3) 代碼運(yùn)行調(diào)試

    這篇文章主要為大家詳細(xì)介紹了最全的Pycharm學(xué)習(xí)教程第三篇代碼運(yùn)行調(diào)試,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-05-05
  • python實(shí)現(xiàn)音樂下載器

    python實(shí)現(xiàn)音樂下載器

    這篇文章主要為大家詳細(xì)介紹了python實(shí)現(xiàn)音樂下載器,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-04-04
  • Python連接es之es更新操作示例詳解

    Python連接es之es更新操作示例詳解

    這篇文章主要為大家介紹了Python連接es之es更新操作示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-05-05
  • Python?提速器numba

    Python?提速器numba

    這篇文章主要介紹了Python?提速器numba,相信大部分人都感嘆過python 真的太好用了,但是它真的好慢啊,然而今天我們就來用numba解決Python?慢的這個(gè)問題,需要的朋友可以參考一下
    2022-01-01
  • python使用json將字符串轉(zhuǎn)字典報(bào)錯(cuò)的解決

    python使用json將字符串轉(zhuǎn)字典報(bào)錯(cuò)的解決

    這篇文章主要介紹了python使用json將字符串轉(zhuǎn)字典報(bào)錯(cuò)的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-02-02
  • Python?pomegranate庫實(shí)現(xiàn)基于貝葉斯網(wǎng)絡(luò)拼寫檢查器

    Python?pomegranate庫實(shí)現(xiàn)基于貝葉斯網(wǎng)絡(luò)拼寫檢查器

    這篇文章主要為大家介紹了Python?pomegranate庫實(shí)現(xiàn)基于貝葉斯網(wǎng)絡(luò)拼寫檢查器示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪<BR>
    2023-04-04

最新評(píng)論