Python 学习笔记 - 装饰器

这一节了解了一些装饰器的简单使用。

首先来看一个预备知识,把一个函数当做参数,传入另外一个函数

比如说我传递outer(f1),我传入的是f1的内存地址,a=func()其实执行了f1()这个函数,并把返回值赋给了a,因此当我打印print(a),他会输出hee

>>> def outer(func):
    print(func)
    a=func()
    print(a)
def f1():
    print("aaa")
    return "hee"
outer(f1)
------------
<function f1 at 0x0000023D3FF3D510>
aaa
hee

装饰器(decorator)就是利用可以把函数当做传递这一点,他可以统一给一些函数添加一些“修饰”功能,但是又不会去破坏原有的代码。装饰器本身也是一个函数,其他函数调用他的时候,在其他函数前面 @ +装饰器名的格式就行了

这个格式执行了2个功能:

1. 自动执行装饰器函数并且将其下面的函数名f1当作参数传递

2. 将装饰器函数的返回值,重复赋值给 f1

简单的说就是这样

f1=decortor(f1)

把上面的例子稍微修改成装饰器,如下所示,结果是一样的

def outer(func):
    def innner(*args,**kwargs):
        print(func)
        a=func(*args,**kwargs)
        print(a)
        return a
    return innner
@outer
def f1():
    print("aaa")
    return "hee"
f1()

再看另外一个复杂一些的例子:

我定义了3个函数f1,f2,f3,他们有不同的参数,我需要每个函数在现在的结果前面添加一个‘before‘,结果后面添加一个‘after’

def outer(func):
    def inner(*args, **kwargs):
        print(‘before‘)
        r = func(*args, **kwargs)
        print(‘after‘)
        return r
    return inner

@outer
def f1(arg):
    print(arg)
    return "F1"
    
@outer
def f2(a1, a2):
    print("F2")
    
@outer
def f3():
    print("F3")
    
    
f1("hhh")
f2("bbb",444)
f3()

结果如下:

before
hhh
after
before
F2
after
before
F3
after

注意要点:

  1. 当使用装饰器outer的时候,他传递参数 outer(f1),这里传递的是f1的地址;r=func()其实执行的就是f1(),然后把返回值赋给了r,这里的目的是为了保证inner返回的结果和f1返回的结果一样;最后再把装饰过的inner函数地址赋给f1,实现装饰的效果;
  2. 记得前面的万能参数 f(*args,**kwargs),可以接受任意的参数

例2:装饰器一个广泛使用的场景是登录验证;很多功能必须判断登录之后才能使用。

下面的例子装饰器通过一个全局变量LOGIN_USER来判断是否已经登录。

#!/usr/bin/env python
# -*- coding:utf-8 -*-

LOGIN_USER = {"is_login": False }
def outer(func):
    def inner(*args, **kwargs):
        if LOGIN_USER[‘is_login‘]:
            r = func()
            return r
        else:
            print("请登录")
    return inner
def outer1(func):
    def inner(*args, **kwargs):
        if LOGIN_USER[‘is_login‘] and LOGIN_USER[‘user_type‘] == 2:
            r = func()
            return r
        else:
            print("请登录,或者权限不够")
    return inner
@outer1
def order():
    print("欢迎%s登录" % LOGIN_USER[‘current_user‘])
@outer
def changepwd():
    print("欢迎%s登录" % LOGIN_USER[‘current_user‘])
@outer
def manager():
    print("欢迎%s登录" % LOGIN_USER[‘current_user‘])
def login(user, pwd):
    if user == "alex" and pwd == "123":
        LOGIN_USER[‘is_login‘] = True
        LOGIN_USER[‘current_user‘] = user
        manager()
def main():
    while True:
        inp = input("1,后台管理;2,登录")
        if inp == ‘1‘:
            manager()
        elif inp == ‘2‘:
            username = input("请输入用户名")
            pwd = input("请输入密码")
            login(username, pwd)
main()

例3.

如果有多个装饰器,我可以嵌套的使用,因为一层的装饰器之后其实也是一个函数,那他自然可以再继续被装饰。

比如说,注意他的调用顺序是从下往上的

Python 3.5.2 (v3.5.2:4def2a2901a5, Jun 25 2016, 22:18:55) [MSC v.1900 64 bit (AMD64)] on win32
>>> def outer(func):
    def innner(*args,**kwargs):
        print(func)
        a=func(*args,**kwargs)
        print("装饰1")
        return a
    return innner
def outer2(func):
    def inner(*args,**kwargs):
        print(func)
        a=func(*args,**kwargs)
        print("装饰2")
        return a
    return inner
@outer2
@outer
def f1():
    print("原函数")
    return "hee"
f1()
<function outer.<locals>.innner at 0x000001FF89A6D620>
<function f1 at 0x000001FF89A6D598>
原函数
装饰1
装饰2

时间: 2024-10-12 17:01:41

Python 学习笔记 - 装饰器的相关文章

python学习笔记--装饰器

1.首先是一个很无聊的函数,实现了两个数的加法运算: def f(x,y): print x+y f(2,3) 输出结果也ok 5 2.可是这时候我们感觉输出结果太单一了点,想让代码的输出多一点看起来不这么单调: def showInfo(fun): def wrap(x,y): print "The function before" func(x,y) print "The function after" return wrap def f(x,y): print

六、PYTHON 学习之装饰器使用

Python是一种强大的语言,即可浅尝辄止,也可深入挖掘.很适合做科学计算.数据挖掘等等.今天我将简单介绍一下Python的装饰器(Decorators)的用法 . 假设我们想要庆祝下生日,需要邀请一些朋友过来参加.但是你有个讨厌的朋友,叫Joe,必须不能让他来啊.可能首先你想到的是建一个list,然后迭代查找并移除所有的Joe童鞋.这当然是个好方法,但是这里为了介绍装饰器,我们会用@来完成这个工作.虽然可能看起来没有什么必要,但是有助于大家学习装饰器的用法. 首先创建一个Python文件app

十、PYTHON 学习之装饰器加深理解

在第六章已经有所介绍,这里看到一篇比较好的文章转过来. 基本概念 装饰器是23z种设计模式之一,经常被用于有切面需求的场景,较为经典的有插入日志.性能测试.事务处理, Web权限校验, Cache等. 很有名的例子来理解,咖啡,加糖的咖啡,加牛奶的咖啡. 本质上,还是咖啡,只是在原有的东西上,做了"装饰",使之附加一些功能或特性. 例如记录日志,需要对某些函数进行记录 笨的办法,每个函数加入代码,如果代码变了,就悲催了 装饰器的办法,定义一个专门日志记录的装饰器,对需要的函数进行装饰.

Python学习之装饰器

装饰器出现的原因是因为我们应对新的需求:在不改变原函数调用方式的情况下,在这个函数前后加上扩展功能:它的本质是闭包函数. 装饰器的结构: def timmer(func): def inner(*args,**kwargs): '''添加函数调用之前的扩展代码''' ret = func(*args,**kwargs) '''添加函数调用之后的扩展代码''' return ret return inner 两个def 两个return互相对应. 装饰器的设计模式:是开放封闭原则:既对扩展是开放的

Python学习之装饰器进阶

函数知识回顾: 函数的参数分为:实参和形参. 实参:调用函数的时候传入的参数: 形参:分为3种(位置参数.默认参数.动态传参) 位置参数:必须传值 def aaa(a,b): print(a,b) aaa(1,2) 默认参数:不是必须传参数 def bbb(x=10): print(x) # bbb() #x = 10 # bbb(20) #x = 20 动态传参 def ccc(*args):#1,2,3,4,5 print(args) ccc(1,2,3,4,5)#按位置传参数 t = (1

python学习之-装饰器

1. 什么是装饰器 装饰器就是用来为被装饰的对象新增功能的工具/函数,被装饰的对象可以是任意可调用的对象,装饰器本身也可以是任意可调用的对象 2. 为何要用装饰器 开放封闭原则:对修改封闭,对扩展开放 装饰器需要遵循的原则: 1. 不能修改被装饰对象的源代码 2. 不能修改被装饰对象的调用方式 PS:装饰器的目标:就是要在遵循原则1和2的前提下,为被装饰对象新增功能 3. 如何实现装饰器 import time def index():  #这是被装饰对象 time.sleep(1) print

python学习-42 装饰器 --- 函数闭包1

函数闭包举例: def father(name): print('hello world') def son(): print('儿子说:我的爸爸是%s' % name) def grandfson(): print('孙子说:我的爷爷是%s' % name) grandfson() son() father('小明') 运行结果: hello world 儿子说:我的爸爸是小明 孙子说:我的爷爷是小明 Process finished with exit code 0 函数的包: 就是嵌套里的

python学习之装饰器的wraps作用

未加@wraps # def tag(name):# def decorator(func):# def wrapper(text):# value=func(text)# return "<{name}>{value}</{name}>".format(name=name,value=value)# return wrapper# return decorator## @tag("p")# def my_upper(text):# valu

python学习:装饰器用法

1 #!/usr/bin/env python 2 #coding:utf-8 3 4 name="alex" 5 psw="abc123" 6 7 def auth(func): 8 def wrapper(*args,**kwargs): 9 username=input('input your username:') 10 password=input('input your password:') 11 12 if username==name and pa