装饰器模式以及在Python下的实现

什么是 开放-封闭原则

开放是对扩展开放,也就是能支持功能的扩展
封闭是对原功能封闭,这意思是当功能扩展的时候并不用更改原来的代码

什么是装饰器模式

装饰器模式就是在不改变原来代码的基础上增加原来方法的功能

举个例子

比如有一个方法叫 method1()

现在需要在method1调用的时候进行日志记录

一般做法是

log()
method1()

但是这样的问题是 所有调用method1()的地方都要添加log()。多处进行更改。

我们需要的是外部的所有调用还是method1()来调用,而method1()执行的时候添加了日志功能,并且还不能更改method1()原来的代码。因为如果要改method1的代码的话,那么后面如果再需要添加一个权限校验功能就又需要改method1的代码。不符合封闭原则。

这个时候就可以用装饰器模式来解决这个问题。

下面以Python语言为例

首先需要说明的是在Python中,函数名其实就是一个指向函数体的一个指针。(应该在所有语言中都是这样的)

即下面这种情况 m2()method1()是一样的。都是指向了同一个方法体

m2 = method1
m2()

如图所示

那么对于方法功能扩展就可以这样来

def w():
    def inner():
        print('log')
    return inner

m = w()

m()

当调用w()时,返回的其实就是inner函数的函数指针

然后m()执行,其实也就是inner()执行。所以就输出log

那么此时w()这个函数和method1还没有产生关系。

现在,我们让inner执行的时候去执行method1


def method1():
    print('method1 do something...')

def w(func):
    def inner():
        print('log')
        func()
    return inner
    
m2 = w(method1)
m2()

现在来解释一下这段代码。

首先是method1的定义

然后是一个w方法的定义,w接受一个参数

接下来调用w,w将内部的inner函数返回。用m2指向inner函数

调用m2也就是调用inner

inner()函数执行,首先输出log,然后执行func(),这个func()方法也就是w()传入的参数。此处即method1

这样最终执行结果就是

先调用日志功能,此处即输出log

再执行method1

但是,外部调用的method1(),如果像上面这样需要都改为m2()。我们需要不影响外部调用 。

如下修改

def method1():
    print('method1 do something...')

def w(func):
    def inner():
        print('log')
        func()
    return inner
    
method1 = w(method1)
method1()

我们让method1重新指向inner函数的地址

这样外部所有的调用还是调用method1() ,但是该方法已经添加了日志功能

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,986评论 19 139
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 32,726评论 18 399
  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 5,875评论 0 9
  • 工厂模式类似于现实生活中的工厂可以产生大量相似的商品,去做同样的事情,实现同样的效果;这时候需要使用工厂模式。简单...
    舟渔行舟阅读 12,396评论 2 17
  • 不懂爱,渴望爱,一直在找寻爱。 我朋友说,我梦到的小红帽,就是一直在找寻的象征。我说,小红帽充满希望,但很容易相信...
    法棍红辣椒阅读 1,148评论 0 0

友情链接更多精彩内容