`
luozhaoyu
  • 浏览: 347615 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

python实现decorator模式

阅读更多
python有个很常用的语法糖是@decorator,使用它可以很方便的创建decorator装饰器模式。(当然,@的用处可不只用在创建装饰器模式)方法有两种,一种是通过创建一个包裹类Wrapper,另一种就是直接通过一个函数创建closure

简要说下装饰器模式:不改变目标的内部行为,改变目标的外在表现方式的一种模式。重点在于,它一定不干涉“内政”,只可以在目标的外围进行修饰,对目标是透明的。

函数的方法最简单
def wrap(num):
    print("wrap initialed")
    def w(func):
        print("w start", num)
        def w2(*args):
            print("w2 start")
            func(*args)
            print("w2 end")
            return "some thing"       
        return w2
    return w

@wrap(10)
def foo2(num, string):
    print(string * num)

print("BEING:")
foo2(3, 'z')
foo2(4, 'y')
print(foo2)

输出结果是:
wrap initialed
('w start', 10)
BEING:
w2 start
zzz
w2 end
w2 start
yyyy
w2 end
<function w2 at 0x7f25d74735f0>

这里就相当与执行了语句
wrap(10)(foo2)(3, 'z')

每一层都先计算出本层的对象,然后再调用被嵌套的函数
这里可以看到
wrap initialed
('w start', 10)

最先输出了一次,说明虽然是包裹函数,但是其已经形成了闭包,全局唯一

如果使用类来实现,情况也比较类似
import datetime
class Wrapper(object):
    def __init__(self, num = -1):
        print("Wrapper Initialed", num)
        self.num = num
        
        
    def __call__(self, func):
        print("Wrapper Called")
        # 以下表明参数的写法虽运行无误,但是是没有意义的
        def w(fff, ggg, *args):
            print("Wrapper Preparing", datetime.datetime.now())
            # 这样传参数是没有意义的
            func(num = ggg, x = self.num,  *args)
            print("Wrapper Finished", datetime.datetime.now())
#        w.__name__ = func.__name__
        return w

#将值3传入Wrapper的self.num
@Wrapper(3)
def foo(num = 1, x = 'z', *args):
    print(x * num, args, 'x=', x, 'num=', num)

    
if __name__ == '__main__':
    print('BEGIN:')
    foo(2, 'a')
    foo(3, 'b')
    print(foo)

输出结果是:
('Wrapper Initialed', 3)
Wrapper Called
BEGIN:
('Wrapper Preparing', datetime.datetime(2012, 5, 5, 0, 25, 58, 136713))
('aaa', (), 'x=', 3, 'num=', 'a')
('Wrapper Finished', datetime.datetime(2012, 5, 5, 0, 25, 58, 136829))
('Wrapper Preparing', datetime.datetime(2012, 5, 5, 0, 25, 58, 136864))
('bbb', (), 'x=', 3, 'num=', 'b')
('Wrapper Finished', datetime.datetime(2012, 5, 5, 0, 25, 58, 136920))
<function w at 0x7f11d4f21758>

可以看出Wrapper类也只执行了一次初始化。我怀疑可不可能是因为
foo = Wrapper(3)(foo)(*args)
返回了一个闭包。通过最后一行的打印结果证实了我的想法。这个类也形成了一个闭包。

值得注意的有两点:一是Wrapper类使用了__call__函数,这个函数仅在发生调用时instance(*args),才会被激活,相当与instance.__call__(*args)
二是不要瞎折腾。 我在这里尝试把Wrapper的参数,传递给foo,这里需要w函数和foo的参数列表保持一致就可以不报错。但是这样做起不到改变foo函数行为的作用,顶多改变这个foo函数的默认参数,所以是毫无意义的!如果没有特殊必要,请不要把Wrapper的参数往foo中塞,Wrapper的函数仅在foo调用前后调用才有意义。

最后一点写参数表的心得体会是,在写装饰器的时候,如果拿不准func参数怎么写,怎么把它包装起来,那么就先写func(*args),然后再完善上面一层封包,直至最后把w(func)函数参数写好。
0
0
分享到:
评论

相关推荐

    python实现Decorator模式实例代码

    本文研究的主要是python实现Decorator模式,具体介绍如下。 一般来说,装饰器是一个函数,接受一个函数(或者类)作为参数,返回值也是也是一个函数(或者类)。首先来看一个简单的例子: # -*- coding: utf-8 -*- ...

    python 实现 状态模式

    - **状态模式3(分类版)**:可能利用了Python的元类(metaclass)或者类别装饰器(class decorator)来动态地改变类的行为,根据对象的状态来切换其方法。 - **状态模式4**:可能引入了更复杂或更优化的实现,比如...

    python使用装饰器(Decorator)的方式实现单例模式

    demo python使用装饰器(Decorator)的方式实现单例模式 functools.wraps 则可以将原函数对象的指定属性复制给包装函数对象, 默认有 __module__、__name__、__doc__,或者通过参数选择

    Python使用logging结合decorator模式实现优化日志输出的方法

    本文实例讲述了Python使用logging结合decorator模式实现优化日志输出的方法。分享给大家供大家参考,具体如下: python内置的loging模块非常简便易用, 很适合程序运行日志的输出。 而结合python的装饰器模式,则可...

    faif python-patterns 使用Python实现一些设计模式的例子

    本资源“faif python-patterns”是针对Python编程语言的一个项目,其中包含了多种设计模式的实现示例。下面,我们将深入探讨这些设计模式及其在Python中的应用。 1. **单例模式(Singleton)**: 单例模式确保一个...

    Python装饰器模式学习demo

    这种设计模式在Python中非常常见,因为它提供了一种灵活的方式来扩展功能,保持代码的整洁和模块化。 在"Python装饰器学习demo"中,我们可以探索以下几个关键知识点: 1. **基础装饰器**:装饰器的基本形式是一个...

    Python-图说设计模式

    在`design_patterns-master`这个压缩包中,可能包含有按照各个设计模式分类的源码文件,每个模式下都有对应的Python实现和相关的图形解释。读者可以逐个研究这些代码示例,结合图形化的解释,加深对设计模式的理解。...

    分析Python中设计模式之Decorator装饰器模式的要点

    Python中的装饰器模式是一种设计模式,它允许在不修改原有对象的情况下向其添加新的功能。装饰器模式在Python中有着广泛的应用,特别是在处理诸如认证、权限检查、日志记录、参数检查、同步加锁等与核心业务逻辑无关...

    Python实现Singleton模式的方式详解

    使用python实现设计模式中的单例模式。单例模式是一种比较常用的设计模式,其实现和使用场景判定都是相对容易的。本文将简要介绍一下python中实现单例模式的几种常见方式和原理。一方面可以加深对python的理解,另一...

    学习python语法,把python的设计模式实现了一遍。.zip

    3. Python实现设计模式的示例: - 使用类和继承实现工厂模式,创建对象时根据条件选择具体的子类。 - 利用__new__和__init__方法实现单例模式,确保每次调用都返回同一实例。 - 定义装饰器函数,包裹原函数,增加...

    设计模式的python实现-py-patterns.zip

    这个"设计模式的python实现-py-patterns.zip"压缩包文件包含了对多种经典设计模式的Python实现。 在Python中,设计模式主要分为三大类:创建型、结构型和行为型。创建型模式关注对象的创建,如单例(Singleton)、...

    装饰器模式[Decorator]

    装饰器模式(Decorator)是一种设计模式,它允许在运行时向对象添加新的行为或责任,而无需修改对象的源代码。这种模式属于结构型模式,是面向对象设计中的一种非常实用的技术。 装饰器模式的核心思想是通过将一个...

    设计模式专题之(七)装饰模式---设计模式装饰模式示例代码(python--c++)

    在Python和C++这两种语言中,装饰模式的实现方式有所不同,但核心思想是一致的。让我们先从Python开始,通过`Decorator.py`文件中的示例来理解装饰模式。 在Python中,装饰器是一种特殊类型的函数,可以用来修改...

    python装饰器decorator介绍

    一、装饰器decorator ... 比较常用的功能一般使用decorator来实现,例如python自带的staticmethod和classmethod。 装饰器有两种形式: 复制代码 代码如下: @A def foo():  pass 相当于: 复制代码 代码如下

    Decorator模式

    在Java或Python等面向对象语言中,Decorator模式通常涉及接口或抽象类,以及一系列装饰者类。装饰者类实现了相同的接口(或继承自相同的抽象类),并且持有一个指向被装饰对象的引用。装饰者可以在调用原始对象的...

    Python-pythonpatterns收集了Python常用的设计模式

    Python的装饰器是实现这一模式的独特方式,它们可以用来修改函数、方法或类的行为。 8. **适配器模式**:将不兼容的接口转换成用户期望的接口,使得原本由于接口不兼容而不能一起工作的类可以协同工作。在Python中...

    Python的设计模式学习案例

    在Python中,这些模式可以通过面向对象特性如继承、封装和多态来实现。例如,单例模式可以通过装饰器实现,而工厂模式则可以利用类的动态性。通过深入理解和应用这些设计模式,开发者可以写出更加灵活、可扩展和易于...

    Python装饰器decorator用法实例

    装饰器是Python中实现装饰模式的一种方式。它是一个可调用的对象,通常是一个函数,接收一个函数作为参数,并返回一个新的函数。装饰器可以用来增强或修改原函数的功能,如日志、性能测试、事务处理等。在示例中,`...

Global site tag (gtag.js) - Google Analytics