什么是 开放-封闭原则
开放是对扩展开放,也就是能支持功能的扩展
封闭是对原功能封闭,这意思是当功能扩展的时候并不用更改原来的代码
什么是装饰器模式
装饰器模式就是在不改变原来代码的基础上增加原来方法的功能
举个例子
比如有一个方法叫 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()
,但是该方法已经添加了日志功能