有参装饰器的学习

通过闭包函数给装饰器传参


这种情况是用于warpper函数的参数不能动,但是我们还要从外部传入一个参数的情况下

我们学过两种传参方式:
1.直接传参
2.闭包函数

以下是我对闭包函数传参的理解,如果可以的话我们尽量不要用闭包函数传参。

我在下面举一个例子:

def login(func):
    def wrapper():
        username = input('输入你的账号')
        password = input('输入你的密码')
        if username == '123' and password == '123':
            func()
        else:
            print('账号密码错误')

    return wrapper


@login
def run():
    print('金屋藏娇')


run()

以上是一个简单的登录认证的装饰器,只有你输入的账号和密码正确了,此函数才能被调用,但是如果我们想在传入一个外部传入一个参数呢,如果我想传入一个webname参数的并且可以做到在全局访问那怎么办,这个时候我们不能在login内部写死,这样的话就改不了了。

那我们应该怎么办,很简单,再给这个装饰器嵌套一层函数就可以了
请看以下的代码:

def web_login(webanme):
    def login(func):
        def wrapper():
            print(f'欢迎来到{webname}')
            username = input('输入你的账号')
            password = input('输入你的密码')
            if username == '123' and password == '123':
                func()
            else:
                print('账号密码错误')
    
        return wrapper
    return login


@web_login('韩信的网站')
def run():
    print('金屋藏娇')

run()

有的同学可能就会说了,老师你突然拿出这么一段代码,我都看蒙蔽了

不急我会分几个步骤给同学们讲解

1. 装饰器原理

我们都知道装饰器可以给一个函数添加功能,增加代码的复用率,但是是怎么实现的呢
我们看回之前的login装饰器代码
def login(func):
    def wrapper():
        username = input('输入你的账号')
        password = input('输入你的密码')
        if username == '123' and password == '123':
            func()
        else:
            print('账号密码错误')

    return wrapper
    
@login
def run():
    print('金屋藏娇')

首先我们看到@这个符号,这其实是一个语法糖
那这里的语法糖会做什么事情呢

他其实相当于一个简写,会让我们的代码更加好看,阅读起来更加美观

我们来看一下没有用语法糖的时候的写法:

def run():
    print('金屋藏娇')
    
run = login(run)

看完之后感觉怎么样,是不是很不美观,但是我们用了语法糖的语句就不一样了在函数定义的上面加上@login就可以做到给函数添加功能

所以说实际上@login等价于 @login ==》 run=login(run)

但其内部的原理是什么呢???
那我们就要来看上面的装饰器定义的代码了
为了方便我们阅读,我先把里面子代码的逻辑代码拿掉先不看

def login(func):
    def wrapper():
            func()
    return wrapper

这里面所做的事情实际上就是把wrapper函数的内存地址赋值给run函数,这样就做到了一个添加功能的作用

上面我们说到了一个语法糖,代码我们拿下来看
@login ==》 run=login(run)
单看一个 run = login(run) 那是不是把login函数的返回值赋值给了run,那我们的login函数的返回值是什么,不就是warpper函数的内存地址吗,那这样是不是我们再调用run函数实则调用的就是warpper函数。

2.为什么们要写再嵌套一层def

通过上面的装饰器原理的讲解,我们了解到了有装饰器的函数实际上是调用装饰器里面的warpper函数,那这样的话我们就可以很简单的理解有参装饰器了,我们再把有参装饰器的代码拿下来看看

def web_login(webanme):
    def login(func):
        def wrapper():
            print(f'欢迎来到{webname}')
            username = input('输入你的账号')
            password = input('输入你的密码')
            if username == '123' and password == '123':
                func()
            else:
                print('账号密码错误')
    
        return wrapper
    return login


@web_login('韩信的网站')
def run():
    print('金屋藏娇')

run()

这里的web_login等价于什么 是不是等价于

outer = web_login('韩信的网站')
@outer
def run():
    print('金屋藏娇')

如果你上面看懂了的话,这里理解起来是不难的,因为调用web_login函数返回的是login的内存地址,我们调用了web_login函数实际上是调用了login函数,顺便把参数给传入了进去,这样的话我们想在外面传入多少个参数都可以了

以上就是一次有参装饰器的讲解,讲的有不足的地方请大家多多包涵

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值