首页 > 编程语言 >Python高频面试题——装饰器(带大家理解装饰器的本质)

Python高频面试题——装饰器(带大家理解装饰器的本质)

时间:2023-09-27 10:40:45浏览次数:40  
标签:-% 面试题 Python kw func print now 装饰 def




Python高频面试题——装饰器(带大家理解装饰器的本质)_打印日志


装饰器概念

装饰器本质上是一个python函数,它可以让其他函数在不需要做任何代码变动的前提下增加额外功能,装饰器的返回值也是一个函数对象。它经常用于有切面需求的场景,比如:插入日志、性能测试、事务处理、缓存、权限验证等场景,装饰器是解决这类问题的绝佳设计,有了装饰器,我们就可以抽离出大量与函数功能本身无关的雷同代码并继续重用。概括的讲,装饰器的作用就是为已经存在的对象添加额外的功能。

装饰器代码实战

友情提示:接下来的代码有点多,希望大家可以拷贝下来,实际运行一下,相信会对装饰器这个概念有更为深刻的理解!

给大家举个例子,定义一个now函数 输出当前时间

def now ():
      print(time.strftime("%Y-%m-%d-%H_%M_%S", time.localtime()))

现在,假设我们要增强now()函数的功能,比如,在函数调用前后自动打印日志,但又不希望修改now()函数的定义,这种在代码运行期间动态增加功能的方式,称之为“装饰器”(Decorator)。

本质上,decorator就是一个返回函数的高阶函数。所以,我们要定义一个能打印日志的decorator,可以定义如下:

def log(func):
     def wrapper(*args, **kw):
            print('call start:' +func.__name__)
            func(*args, **kw)
            print('call end:' + func.__name__)
      return wrapper

这是一个较为固定的写法:

参数func表示传入的函数对象

wrapper是内部函数,return wrapper 会实现对其调用

(*args, **kw)是一种固定用法,表示可以传入任意的参数,*args和**kw分别属于非关键字参数和关键字参数,两者也都是可变参数。一个星号*加上参数名,比如*number,定义后,number可以接收任意数量的参数,并将它们储存在一个tuple中。关键字参数的特征是两个星号**加上参数名,比如**kw, 定义后,kw将接收到的任意数量参数存到一个dict中。举个例子就懂了

def func_para(*args, **kw):
    print ('args:',args )
    print ('kw:',kw )

func_para(1,2,3,4, a=1,b=2,c=3)

输出:

args: (1, 2, 3, 4)

kw: {'a': 1, 'b': 2, 'c': 3}

func(*args, **kw) 表示对传入的函数进行调用,调用前后分别执行了两条print语句

func.__name__ 表示函数的名字

def wrapper 根据需求也可以return 某个值。

最后调用装饰器的代码如下:在now上面加上装饰器 @log即可

def func_para(*args, **kw):
    print ('args:',args )
    print ('kw:',kw )

func_para(1,2,3,4, a=1,b=2,c=3)
@log
def now ():
      print(time.strftime("%Y-%m-%d-%H_%M_%S", time.localtime()))
now()

输出:

call start:now

2023-03-10-15_08_40

call end:now

看到这里有的同学可能会问,如果now()函数需要增加参数怎么办?很简单,我们无须对装饰器log进行任何修改,代码如下:

@log
def now (a,b):
      print(a)
      print(time.strftime("%Y-%m-%d-%H_%M_%S", time.localtime()))
      print(b)
now("test1","test2")

输出:

call startnow

test1

2023-03-10-15_08_40

test2

call end:now

装饰器的本质

其实想要了解装饰器的本质,我们需要了解python的函数对象!python中一切皆是对象,所以函数也不例外,我们还是以下面代码为例

def now ():
     print(time.strftime("%Y-%m-%d-%H_%M_%S", time.localtime()))
print(now)
print(type(now))

输出:

<function now at 0x000001C7C5D44F78>

<class 'function'>

可以看到输出了now对象的地址和对应的类型。我们也可以理解函数的名字就是函数在内存中对应的地址

我们可以把函数赋值:

a=now

print(a)

此时输出的a 的值与print(now) 是一样的!

我们也可以把函数作为参数传递,例如

import time
def now():
     print(time.strftime("%Y-%m-%d-%H_%M_%S", time.localtime()))

def func_demo(func):
     return func #调用作为参数传入的函数

func1=func_demo(now)
func1()

输出:

2023-03-10-15_18_44

讲到这里,我们可以看出来

@log

def now ():

其实等价于

log(now)

这里now作为了装饰器函数log的参数,@log只是语法糖而已,语法糖是计算机语言中特殊的某种语法,这种语法对语言的功能并没有影响,对于程序员有更好的易用性,能够增加程序的可读性。大家可以结合前面讲解的def log函数的代码,然后执行以下代码

import time
def log(func):
    def wrapper(*args, **kw):
        print('call start:' + func.__name__)
        func(*args, **kw)
        print('call end:' + func.__name__)

    return wrapper
@log
def now ():
   print(time.strftime("%Y-%m-%d-%H_%M_%S", time.localtime()))

d=log(now)
d()

会输出:

call start:wrapper

call start:now

2023-03-10-15_29_47

call end:now

call end:wrapper

标签:-%,面试题,Python,kw,func,print,now,装饰,def
From: https://blog.51cto.com/liwen629/7621109

相关文章

  • Python高频面试题——生成器(最通俗的讲解)
    生成器定义在Python中,使用了yield的函数被称为生成器(generator)。跟普通函数不同的是,生成器是一个返回迭代器的函数,只能用于迭代操作,更简单点理解生成器就是一个迭代器。在调用生成器运行的过程中,每次遇到yield时函数会暂停并保存当前所有的运行信息,返回yield的值,并在......
  • playwright python环境运行报错 ImportError: DLL load failed
    网上建议网上好多文章介绍playwright的环境搭建,用以下两条语句即可完成pipinstallplaywrightplaywrightinstall安装完毕后,尝试执行一段经典的pythondemofromplaywright.sync_apiimportsync_playwrightwithsync_playwright()asp:browser=p.chromium.launch()page......
  • Python高频面试题——迭代器和可迭代对象
     无论是面试测试还是运维涉及到python编码岗位时,迭代器和可迭代对象都是绕不开的一个问题,本文对这两个概念进行重点讲解,本文从什么是迭代讲起,然后介绍迭代器和可迭代对象二者的区别,最后通过for循环和自定义迭代器来加深读者对这两个概念的理解,只要认真阅读完文章,相信一定会帮助到......
  • Python爬虫-爬取百度搜索结果页的网页标题及其真实网址
    共两个依赖的需提前安装的第三方库:requests和bs4库cmd命令行输入安装requests库:pip3install-ihttps://pypi.douban.com/simplerequests安装bs4库:pip3install-ihttps://pypi.douban.com/simplebeautifulsoup4 本微项目源文件下载地址:https://wwuw.lanzouj.com/i1Au51......
  • Python 图片并行下载
    需求:有大量图片的url需要将其快速下载到本地技术点:采用编写并发代码的库asyncio以及基于asyncio实现的HTTP框架aiohttppipinstallasynciopipinstallaiohttp代码如下:importjsonimportosimportrequestsimportaiohttpimportasyncioimage_save_dir="images"......
  • Python面试高频问题:修改list中某个元素时的坑
    在Python面试中经常会考这样一个题目,遍历列表,如果列表中有某某元素,那么将其替换成"test"。题目看似简单,其实有个坑在里面!从面试结果来看,大多数同学都会这样写:l=["a","b","c"]foriinl:if"a"==i:i="test"print(l)运行后,大家会发现输出的l值还是['a',......
  • python range中的步长必须是整数 numpy则可以是小数
    >>>foriiinrange(1,10,0.1): print(ii)Traceback(mostrecentcalllast):File"<pyshell#4>",line1,in<module>foriiinrange(1,10,0.1):TypeError:'float'objectcannotbeinterpretedasaninteger>>......
  • Python脚本连接Oracle数据库并验证成功
    #yaml文件存储数据->root\Data\oracle_admin_f_shozaiko.yaml#TestDataforOracleDB:ADMIN->F_SHOZAIKO-name:connecttoOraclerequest:uname:adminupwd:P823!ApoLhost:rf-oms.cbfvvrud0bld.ap-northeast-1.rds.amazonaws.com:1521/rfomsqu......
  • python学习框架
    Python简介与安装Python的历史与特点Python的安装与配置Python基础语法变量与数据类型运算符与表达式控制结构(条件判断与循环)函数与模块错误处理与异常Python数据结构列表(List)元组(Tuple)集合(Set)字典(Dictionary)Python面向对象编程类与对象继承与多态......
  • Python的Selenium库:鼠标滚动和操作弹出窗口
    Selenium是一个用于自动化web应用测试的开源工具。通过Selenium,我们可以模拟真实用户的操作,如点击、输入、滚动页面等,来测试web应用的稳定性和可靠性。PythonSelenium库是Selenium的一个分支,可以方便地与Python语言结合使用。在PythonSelenium库中,元素定位和文本输入是最常用的......