python3 装饰器

简介: 内嵌函数def outer(): x = 1 def inner(): print('In inner') print(x) print(locals()) inner() # 若inner调用没有写在这里,inner函数仅仅被声明b=outer()outputs:In inner1{'x': 1}以上代码看起来有些复杂,但它仍是易于理解的。

内嵌函数

def outer():
    x = 1
    def inner():
        print('In inner')
        print(x)
        print(locals())
    inner()     # 若inner调用没有写在这里,inner函数仅仅被声明

b=outer()

outputs:
In inner
1
{'x': 1}

以上代码看起来有些复杂,但它仍是易于理解的。来看 #1 —— Python 搜索局部变量 x 失败,然后在属于另一个函数的外层作用域里寻找。变量 x 是函数 outer 的局部变量,但函数 inner 仍然有外层作用域的访问权限(至少有读和修改的权限)。在 #2 处调用函数 inner。值得注意的是,inner 在此处也只是一个变量名,遵循 Python 的变量查找规则——Python 首先在 outer 的作用域查找并找到了局部变量 inner。

函数是 Python 中的一级对象

在 Python 中有个常识:函数和其他任何东西一样,都是对象。函数包含变量,它并不那么特殊。

也许你从未考虑过函数可以有属性——但是函数在 Python 中,和其他任何东西一样都是对象。(如果对此感觉困惑,稍后你会看到 Python 中的类也是对象,和其他任何东西一样!)也许这有点学术的感觉——在 Python 中函数只是常规的值,就像其他任意类型的值一样。这意味着可以将函数当做实参传递给函数,或者在函数中将函数作为返回值返回。如果你从未想过这样使用,请看下面的可执行代码:

# 函数作为参数,因为在python中函数也是普通的对象,和类对象,变量一样
def add(x,y):
    return x + y

def sub(x,y):
    return x - y
def apply(func, x, y):
    return func(x,y)

print(apply(add,1,5))   # 6

和其他变量一样,函数名就是变量标签.

再来看下面的例子:

def outer1():
    def inner():
        print('Inside inner')
    return inner    #1

f1=outer1() #2
f1()

outputs: Inside inner

这看起来也许有点怪异。在 #1 处返回一个其实是函数标签的变量 inner。也没有什么特殊语法——函数 outer 1返回了并没有被调用的函数 inner。还记得变量的生命周期吗?每次调用函数 outer 的时候,函数 inner 会被重新定义,但是如果函数 outer1 没有返回 inner,当 inner 超出 outer 的作用域,inner 的生命周期将结束。在 #2 处将获得返回值即函数 inner,并赋值给新变量 foo。

闭包

请参考专门的一片关于闭包的文章

>>> def outer(x):
...     def inner():
...         print x # 1
...     return inner
>>> print1 = outer(1)
>>> print2 = outer(2)
>>> print1()
1
>>> print2()
2

装饰器

装饰器其实就是一个以函数作为参数并返回一个替换函数的可执行函数。让我们从简单的开始,直到能写出实用的装饰器。

def outer2(some_func):
    def inner():
        print('before func:')
        ret=some_func()
        return ret + 1    #1
    return inner # 注意不要带括号

def foo2():
    return 1

decorated = outer2(foo2) # 注意是不带()的foo2 #2
print(decorated())

outputs: 
before func:
2

请仔细看这个装饰器示例。首先,定义了一个带单个参数 some_func 的名为 outer 的函数。然后在 outer 内部定义了一个内嵌函数 inner。inner 函数将打印一行字符串然后调用 some_func,并在 #1 处获取其返回值。在每次 outer 被调用时,some_func 的值可能都会不同,但不论 some_func 是什么函数,都将调用它。最后,inner 返回 some_func() 的返回值加 1。在 #2 处可以看到,当调用赋值给 decorated 的返回函数时,得到的是一行文本输出和返回值 2。

我们可以说变量 decorated 是 foo 的装饰版——即 foo 加上一些东西。

class Coordinate:
    def __init__(self,x,y):
        self.x=x
        self.y=y
    def __repr__(self):
        return 'Coord: ' + str(self.__dict__)

a=Coordinate(3,4)
print(a)

outputs: Coord: {'x': 3, 'y': 4}

接着改写

class Coordinate:
    def __init__(self,x,y):
        self.x=x
        self.y=y
    def __repr__(self):
        return 'Coord: ' + str(self.__dict__)

# a=Coordinate(3,4)
# print(a)

def add2(a,b):
    return Coordinate(a.x+b.x, a.y+b.y)

def sub2(a,b):
    return Coordinate(a.x - b.x, a.y - b.y)

one = Coordinate(100, 200)
two = Coordinate(300, 200)
print(add2(one, two))
print(sub2(one, two))
# Coord: {'y': 400, 'x': 400}
# Coord: {'y': 0, 'x': -200}

函数装饰器 @ 符号的应用

@符号是装饰器的语法糖,在定义函数的时候使用

@bar  
def foo():  
    print "foo"  

等价于

def foo():  
    print "foo"  
foo = bar(foo) 

无参数装饰器

def foo(func):  
    print 'decorator foo'  
    return func  

@foo  
def bar():  
    print 'bar'  

bar() 

foo 函数被用作装饰器,其本身接收一个函数对象作为参数,然后做一些工作后,返回接收的参数,供外界调用。
注意: 时刻牢记 @foo 只是一个语法糖,其本质是 foo = bar(foo)

def logger(func):
    def inner(*args, **kwargs): #1
        print("Arguments were: %s, %s" % (args, kwargs))
        return func(*args, **kwargs) #2
    return inner

@logger
def foo1(x, y=1):
    return x * y
@logger
def foo2():
    return 2

rfoo1 = foo1(5, 4)  #Arguments were: (5, 4), {
print(rfoo1)    # 20

print(foo1(1))
# Arguments were: (1,), {}
# 1

带参数的装饰器

def use_logging(level):
    def decorator(func):
        def wrapper(*args,**kwargs):
            if level == 'warn':
                print('%s is running' % func.__name__)
            return func(*args)
        return wrapper
    return decorator

@use_logging(level='warn')
def foo3(name='foo3'):
    print('I am foo3')

foo3()
# foo3 is running
# I am foo3

@use_logging(level=’warn’) 调用的时候,python能够发现这一层的封装,并把参数传递到装饰器中

类装饰器

class Foo():
    def __init__(self,func):
        self._func=func

    def __call__(self, *args, **kwargs):
        print('class deco running')
        self._func()
        print('class deco ending')

@Foo
def bars():
    print('Bars')

bars()
# class deco running
# Bars
# class deco ending

所有的函数都是可调用对象。

一个类实例也可以变成一个可调用对象,只需要实现一个特殊方法call()。

class Person(object):
    def __init__(self, name, gender):
        self.name = name
        self.gender = gender

    def __call__(self, friend):
        print ('My name is %s...' % self.name)
        print ('My friend is %s...' % friend)
p = Person('Bob', 'male')
print(p('Tim'))

# My name is Bob...
# My friend is Tim...

简单来讲就是将动态输入的函数转成内嵌函数所需要表达的内容

# -*- coding:gbk -*-
'''示例6: 对参数数量不确定的函数进行装饰,
参数用(*args, **kwargs),自动适应变参和命名参数'''


def deco(func):
    def _deco(*args, **kwargs):
        print("before %s called." % func.__name__)
        ret = func(*args, **kwargs)
        print("  after %s called. result: %s" % (func.__name__, ret))
        return ret

    return _deco


@deco
def myfunc(a, b):
    print(" myfunc(%s,%s) called." % (a, b))
    return a + b


@deco
def myfunc2(a, b, c):
    print(" myfunc2(%s,%s,%s) called." % (a, b, c))
    return a + b + c


myfunc(1, 2)
myfunc(3, 4)
myfunc2(1, 2, 3)
myfunc2(3, 4, 5)

# before myfunc called.
#  myfunc(1,2) called.
#   after myfunc called. result: 3
# before myfunc called.
#  myfunc(3,4) called.
#   after myfunc called. result: 7
# before myfunc2 called.
#  myfunc2(1,2,3) called.
#   after myfunc2 called. result: 6
# before myfunc2 called.
#  myfunc2(3,4,5) called.
#   after myfunc2 called. result: 12

args 和 *kwargs

定义函数时使用 * 的用法意味着任何传递给函数的额外位置参数都是以 * 开头的

>>> def one(*args):
...     print args # 1
>>> one()
()
>>> one(1, 2, 3)
(1, 2, 3)
>>> def two(x, y, *args): # 2
...     print x, y, args
>>> two('a', 'b', 'c')
a b ('c',)

星* 符号也可以用在函数调用时,在这里它也有类似的意义。在调用函数时,以 * 开头的变量表示该变量内容需被取出用做位置参数。再举例如下:

>>> def add(x, y):
...     return x + y
>>> lst = [1,2]
>>> add(lst[0], lst[1]) # 1
3
>>> add(*lst) # 2
3

在 #1 处的代码和 #2 处的作用相同——可以手动做的事情,在 #2 处 Python 帮我们自动处理了。这看起来不错,*args 可以表示在调用函数时从迭代器中取出位置参数, 也可以表示在定义函数时接收额外的位置参数。

接下来介绍稍微复杂一点的用来表示字典和键值对的 *,就像 用来表示迭代器和位置参数。

>>> def foo(**kwargs):
...     print kwargs
>>> foo()
{}
>>> foo(x=1, y=2)
{'y': 2, 'x': 1}

当定义一个函数时,使用 kwargs 来表示所有未捕获的关键字参数将会被存储在字典 kwargs 中。此前 args 和 kwargs 都不是 Python 中语法的一部分,但在函数定义时使用这两个变量名是一种惯例。和 * 的使用一样,可以在函数调用和定义时使用

>>> dct = {'x': 1, 'y': 2}
>>> def bar(x, y):
...     return x + y
>>> bar(**dct)

调用顺序

参考:http://python.jobbole.com/85056/
http://www.cnblogs.com/rhcad/archive/2011/12/21/2295507.html

目录
相关文章
|
1月前
|
设计模式 前端开发 Shell
Python装饰器是什么?
装饰器是Python中用于动态修改函数、方法或类功能的工具,无需改变原代码。通过将函数作为参数传递并返回新函数,装饰器可以在原函数执行前后添加额外逻辑。例如,使用`@logger`装饰器可以打印函数调用日志,而`@timethis`则可用于计算函数执行时间。为了保持被装饰函数的元信息(如`__name__`和`__doc__`),可使用`functools.wraps`装饰器。此外,带参数的装饰器可通过嵌套函数实现,如`@timeitS(2)`,以根据参数条件输出特定信息。
90 59
|
1月前
|
测试技术 数据库 Python
Python装饰器实战:打造高效性能计时工具
在数据分析中,处理大规模数据时,分析代码性能至关重要。本文介绍如何使用Python装饰器实现性能计时工具,在不改变现有代码的基础上,方便快速地测试函数执行时间。该方法具有侵入性小、复用性强、灵活度高等优点,有助于快速发现性能瓶颈并优化代码。通过设置循环次数参数,可以更准确地评估函数的平均执行时间,提升开发效率。
106 61
Python装饰器实战:打造高效性能计时工具
|
2月前
|
缓存 数据安全/隐私保护 Python
python装饰器底层原理
Python装饰器是一个强大的工具,可以在不修改原始函数代码的情况下,动态地增加功能。理解装饰器的底层原理,包括函数是对象、闭包和高阶函数,可以帮助我们更好地使用和编写装饰器。无论是用于日志记录、权限验证还是缓存,装饰器都可以显著提高代码的可维护性和复用性。
50 5
|
2月前
|
存储 缓存 Python
Python中的装饰器深度解析与实践
在Python的世界里,装饰器如同一位神秘的魔法师,它拥有改变函数行为的能力。本文将揭开装饰器的神秘面纱,通过直观的代码示例,引导你理解其工作原理,并掌握如何在实际项目中灵活运用这一强大的工具。从基础到进阶,我们将一起探索装饰器的魅力所在。
|
2月前
|
Python
探索Python中的装饰器:简化代码,增强功能
在Python的世界里,装饰器就像是给函数穿上了一件神奇的外套,让它们拥有了超能力。本文将通过浅显易懂的语言和生动的比喻,带你了解装饰器的基本概念、使用方法以及它们如何让你的代码变得更加简洁高效。让我们一起揭开装饰器的神秘面纱,看看它是如何在不改变函数核心逻辑的情况下,为函数增添新功能的吧!
|
2月前
|
测试技术 开发者 Python
探索Python中的装饰器:从入门到实践
装饰器,在Python中是一块强大的语法糖,它允许我们在不修改原函数代码的情况下增加额外的功能。本文将通过简单易懂的语言和实例,带你一步步了解装饰器的基本概念、使用方法以及如何自定义装饰器。我们还将探讨装饰器在实战中的应用,让你能够在实际编程中灵活运用这一技术。
50 7
|
2月前
|
程序员 测试技术 数据安全/隐私保护
深入理解Python装饰器:提升代码重用与可读性
本文旨在为中高级Python开发者提供一份关于装饰器的深度解析。通过探讨装饰器的基本原理、类型以及在实际项目中的应用案例,帮助读者更好地理解并运用这一强大的语言特性。不同于常规摘要,本文将以一个实际的软件开发场景引入,逐步揭示装饰器如何优化代码结构,提高开发效率和代码质量。
74 6
|
2月前
|
测试技术 开发者 Python
深入理解Python装饰器:从基础到高级应用
本文旨在为读者提供一个全面的Python装饰器指南,从其基本概念讲起,逐步深入探讨其高级应用。我们将通过实例解析装饰器的工作原理,并展示如何利用它们来增强函数功能、控制程序流程以及实现代码的模块化。无论你是Python初学者还是经验丰富的开发者,本文都将为你提供宝贵的见解和实用的技巧,帮助你更好地掌握这一强大的语言特性。
75 4
|
2月前
|
开发者 Python
探索Python中的装饰器:从基础到高级应用
本文将带你深入了解Python中的装饰器,这一强大而灵活的工具。我们将一起探讨装饰器的基本概念,它们如何工作,以及如何使用它们来增强函数和类的功能,同时不改变其核心逻辑。通过具体代码示例,我们将展示装饰器的创建和使用,并探索一些高级应用,比如装饰器堆栈和装饰带参数的装饰器。无论你是初学者还是有经验的开发者,这篇文章都将为你提供新的视角,帮助你更有效地使用装饰器来简化和优化你的代码。
|
2月前
|
测试技术 数据安全/隐私保护 开发者
探索Python中的装饰器:从基础到高级应用
装饰器在Python中是一个强大且令人兴奋的功能,它允许开发者在不修改原有函数代码的前提下增加额外的功能。本文将通过具体代码示例,带领读者从装饰器的基础概念入手,逐步深入到高级用法,如带参数的装饰器和装饰器嵌套等。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的见解和技巧。

热门文章

最新文章

推荐镜像

更多