python--装饰器

简介: python--装饰器

前戏


装饰器:

定义:本质是函数,(装饰其他函数)就是为其他函数添加附加功能

原则:

  • 不能修改被装饰的函数的源代码
  • 不能修改被装饰的函数的调用方式

实现装饰器知识储备

  1.函数即'变量'

  2.高阶函数

  • 把一个函数当做实参传递给另外一个参数(在不修改被装饰函数源代码的情况下为其添加功能)
  • 返回值中包含函数名(不修改函数的调用方式)

  3.嵌套函数(在一个函数的函数体内用def声明另一个函数)


无参数的装饰器


下面有一个函数,如果我们想统计一下函数执行的时间,使用装饰器怎么实现呢

import time
def test1():
    time.sleep(2)
    print('in the test1')

如果不使用装饰器我们可以在之前前加一个时间,执行后加一个时间就可以了,如下

import time
def test1():
    start_time = time.time()
    time.sleep(2)
    print('in the test1')
    print(time.time()-start_time)
test1()

但是这样的话,我们修改了源代码,如果这个函数有好多地方调用了,有些地方不希望要其他的功能,有些地方需要,难道我们需要重新写一个函数吗?在把调用到的地方都修改一下?如果你这样的话,明天就可以去财务领工资了。。。那来看看怎么使用装饰器来实现

1 import time
 2
 3
 4 def timmer(func):
 5     def warpper():
 6         start_time = time.time()
 7         func()  # run test1
 8         stop_time = time.time()
 9         print('the func time is %s' % (stop_time - start_time))
10
11     return warpper
12
13
14 @timmer  # test1=timmer(test1)语法糖,给哪个函数加装饰器,就在哪个函数前加上 @ 装饰名
15 def test1():
16     time.sleep(2)
17     print('in the test1')
18
19 test1()  # 运行warpper()

代码解释:

函数从上往下执行,执行到14行的时候,14行的代码等价于test1=timmer(test1)。就回去执行timmer函数,所以timmer里的func就是test1。执行完返回了warpper,所以test1=warpper,然后执行19行,因为test1=warpper,所以19行相当于warpper(),执行了waepper函数,而func为test1,func()就是调用了test1()这个函数。

这样我们就实现了一个装饰器,在不修改源代码和调用方式的情况下给函数增加了新的功能。


带参数的装饰器


上面的test1函数没有参数,如果test1需要一个参数,还这样写的话就会报错

import time
def timmer(func):
    def warpper():
        start_time = time.time()
        func()  # run test1
        stop_time = time.time()
        print('the func time is %s' % (stop_time - start_time))
    return warpper
@timmer  
def test1(m1):
    time.sleep(2)
    print('in the test1')
test1('2')

结果:

TypeError: warpper() takes 0 positional arguments but 1 was given

报错信息告诉了我们warpper需要一个参数,那我们去修改一下

import time
def timmer(func):
    def warpper(arg):
        start_time = time.time()
        func(arg)  # run test1
        stop_time = time.time()
        print('the func time is %s' % (stop_time - start_time))
    return warpper
@timmer
def test1(m1):
    time.sleep(2)
    print('in the test1')
test1('2')

我们给warpper传了一个参数arg,因为func(test1)是需要一个参数的,那如果test1需要多个参数怎么办呢,我们可以用可变参数*args,**kwargs来实现动态的传参

import time
def timmer(func):
    def warpper(*args, **kwargs):
        start_time = time.time()
        func(*args, **kwargs)  # run test1
        stop_time = time.time()
        print('the func time is %s' % (stop_time - start_time))
    return warpper
@timmer
def test1(m):
    time.sleep(2)
    print('in the test1')
test1('12')

这样不管test1有没有参数,有几个参数都能满足需求


有返回值的装饰器


还是上面的代码,假如test1里有返回值,如下

1 import time
 2
 3
 4 def timmer(func):
 5     def warpper(*args, **kwargs):
 6         start_time = time.time()
 7         func(*args, **kwargs)  # run test1
 8         stop_time = time.time()
 9         print('the func time is %s' % (stop_time - start_time))
10
11     return warpper
12
13
14 @timmer
15 def test1():
16     time.sleep(2)
17     print('in the test1')
18     return 666
19
20 # test1()
21 print(test1())

结果:

in the test1
the func time is 2.0003857612609863
None

我们可以看到,结果返回了一个None,因为warpper函数的时候,在执行到func(*args, **kwargs)的时候,执行了test1函数,虽然是执行了,但是return返回的值没有接收,那我们可以定义一个变量接收一下,然后再return给调用者

1 import time
 2
 3
 4 def timmer(func):
 5     def warpper(*args, **kwargs):
 6         start_time = time.time()
 7         res = func(*args, **kwargs)  # run test1
 8         stop_time = time.time()
 9         print('the func time is %s' % (stop_time - start_time))
10         return res
11
12     return warpper
13
14
15 @timmer
16 def test1():
17     time.sleep(2)
18     print('in the test1')
19     return 666
20
21 # test1('12')
22 print(test1())

这样的话,我们也实现了一个带返回值的装饰器,如果test1没有返回值,则返回的是None,在看个栗子,加深下理解

def auth(func):
    def timer(*args, **kwargs):
        print('加的新功能start。。。')
        res = func(*args, **kwargs)  # 将返回值得结果赋给res
        print('加的新功能end。。。')
        return res  # 执行返回值的结果,没有不执行
    return timer
@auth  # test2=auth(test2)
def test2(*args, **kwargs):
    print('in the test2...')
    return '我是返回值'
print(test2('zou', 'IT', '24'))

结果:

加的新功能start。。。
in the test2...
加的新功能end。。。
我是返回值


实战


1 user, passwd = 'zou', '123'
 2
 3 def auth(func):
 4     def timmer(*args, **kwargs):
 5         username = input('输入用户名:')
 6         password = input('输入密码:')
 7         if username == user and password == passwd:
 8             print('加的新功能')
 9             res = func(*args, **kwargs)  # 如果不将结果赋给res就没有返回值
10             return res
11         else:
12             exit('密码错误')
13     return timmer
14
15 def test1():
16     print('in the test1')
17
18
19 @auth
20 def test2(name, job):
21     print('name:%s' % name)
22     print('job:%s' % job)
23
24
25 @auth
26 def test3():
27     print('in the test3')
28     return 'end'
29
30
31 test1()
32 test2('zou', 'IT')
33 print(test3())

大家可以自己在脑海里想一下,是怎么运行的,为什么这么运行?最后运行的结果是什么?

结果:

点我偷看


相关文章
|
5天前
|
Python
深入理解Python装饰器:从入门到实践####
本文旨在通过简明扼要的方式,为读者揭开Python装饰器的神秘面纱,从基本概念、工作原理到实际应用场景进行全面解析。不同于常规的摘要仅概述内容概要,本文将直接以一段精炼代码示例开篇,展示装饰器如何优雅地增强函数功能,激发读者探索兴趣,随后深入探讨其背后的机制与高级用法。 ####
33 11
|
6天前
|
设计模式 Python
掌握Python中的装饰器
【10月更文挑战第34天】装饰器是Python中一种强大的工具,它允许我们在不修改原函数代码的情况下增加其功能。本文通过简单易懂的语言和实例,引导你理解装饰器的概念、种类及其应用,帮助你在编程实践中灵活使用这一高级特性。
|
2天前
|
设计模式 缓存 开发者
深入浅出Python装饰器
【10月更文挑战第39天】本文将通过浅显易懂的语言和生动的比喻,带你探索Python中一个神奇而又强大的特性——装饰器。我们将一起揭开装饰器的神秘面纱,了解它的工作原理,并通过实际代码示例学习如何应用它来美化我们的代码。无论你是编程新手还是有经验的开发者,这篇文章都将为你打开一扇新的大门,让你的代码更加优雅和高效。
|
2天前
|
缓存 监控 测试技术
Python中的装饰器:功能扩展与代码复用的利器###
本文深入探讨了Python中装饰器的概念、实现机制及其在实际开发中的应用价值。通过生动的实例和详尽的解释,文章展示了装饰器如何增强函数功能、提升代码可读性和维护性,并鼓励读者在项目中灵活运用这一强大的语言特性。 ###
|
6天前
|
缓存 开发者 Python
探索Python中的装饰器:简化代码,增强功能
【10月更文挑战第35天】装饰器在Python中是一种强大的工具,它允许开发者在不修改原有函数代码的情况下增加额外的功能。本文旨在通过简明的语言和实际的编码示例,带领读者理解装饰器的概念、用法及其在实际编程场景中的应用,从而提升代码的可读性和复用性。
|
1天前
|
设计模式 缓存 开发框架
Python中的装饰器:从入门到实践####
本文深入探讨了Python中装饰器的工作原理与应用,通过具体案例展示了如何利用装饰器增强函数功能、提高代码复用性和可读性。读者将学习到装饰器的基本概念、实现方法及其在实际项目开发中的实用技巧。 ####
11 3
|
2天前
|
缓存 测试技术 数据库
深入理解Python中的装饰器
在本文中,我们将探讨Python语言中一个强大而灵活的特性——装饰器。装饰器允许开发者在不修改原有函数或方法代码的情况下增加额外的功能,这大大提高了代码的复用性和可读性。通过具体示例和应用场景的讲解,本篇文章旨在为读者提供一个关于如何使用装饰器的全面指南,包括装饰器的定义、使用场景、以及如何自定义装饰器等内容。
|
2天前
|
Python
探索Python中的装饰器:简化代码,提升效率
【10月更文挑战第39天】在编程的世界中,我们总是在寻找使代码更简洁、更高效的方法。Python的装饰器提供了一种强大的工具,能够让我们做到这一点。本文将深入探讨装饰器的基本概念,展示如何通过它们来增强函数的功能,同时保持代码的整洁性。我们将从基础开始,逐步深入到装饰器的高级用法,让你了解如何利用这一特性来优化你的Python代码。准备好让你的代码变得更加优雅和强大了吗?让我们开始吧!
7 1
|
7天前
|
设计模式 缓存 监控
Python中的装饰器:代码的魔法增强剂
在Python编程中,装饰器是一种强大而灵活的工具,它允许程序员在不修改函数或方法源代码的情况下增加额外的功能。本文将探讨装饰器的定义、工作原理以及如何通过自定义和标准库中的装饰器来优化代码结构和提高开发效率。通过实例演示,我们将深入了解装饰器的应用,包括日志记录、性能测量、事务处理等常见场景。此外,我们还将讨论装饰器的高级用法,如带参数的装饰器和类装饰器,为读者提供全面的装饰器使用指南。
|
3天前
|
存储 缓存 监控
掌握Python装饰器:提升代码复用性与可读性的利器
在本文中,我们将深入探讨Python装饰器的概念、工作原理以及如何有效地应用它们来增强代码的可读性和复用性。不同于传统的函数调用,装饰器提供了一种优雅的方式来修改或扩展函数的行为,而无需直接修改原始函数代码。通过实际示例和应用场景分析,本文旨在帮助读者理解装饰器的实用性,并鼓励在日常编程实践中灵活运用这一强大特性。