标签:通用 灵活 操作 %s 接收 函数赋值 内聚 pre 结构
作用域是程序运行时变量可以被引用的范围。
在函数内部可以访问全局变量,在函数外部不能访问局部变量。
把一个函数定义在另外一个函数的内部,就是函数嵌套。外边的函数为外层函数,里边的函数为内层函数。
在函数嵌套中,内层函数对外层函数的局部变量进行了引用,并且外层函数的返回值是内层函数的引用,就构成了一个闭包。
def outer(a, b):
a = a
b = b
def inside():
print(a+b)
return inside
function_inside = outer(10, 20)
function_inside()
运行结果:
30
修改外层函数的变量使用nonlocal关键字
闭包函数必须返回内层函数的引用
闭包就是函数和函数独有的数据 结合在一起,它比类要轻量级,比一般的函数功能更强。
每次调用外层函数都会创建一个新的闭包对象。
内层函数没有引用的变量会在外层函数执行结束后销毁。
而我们需要给函数增加新的功能,又不想改变函数的代码的时候,便用到了 装饰器
python函数可以被当作参数传递给其他函数。
def say(func):
func()
def test():
print("this is a function")
say(test)
运行结果:
this is a function
python的函数可以像变量一样作为返回值返回,被定义在函数内部,而且还可以作为参数被传递。
原有函数:
def test():
print("this is a function")
现在需要给函数增加另外一个功能,限制该函数的调用。
print("use limit")
我们可以直接把功能加入到函数内部:
def test():
print("use limit")
print("this is a function")
test()
但是如果有函数test1、test2()...也想使用该功能,我们就需要一一修改,这就造成了:
我们可以定义一个新的函数,来完成新增加的功能,把原有函数当作参数传入新函数执行。
def limit(func):
print("use limit")
func()
def test():
print("this is a function")
limit(test)
这样不仅没修改原有的函数,而且增加了新的功能,但是也存在新的问题:
test
而是调用新函数 limit
test
函数的时候都需要调用 limit
,如果有已存在的调用,将无法使用新加入的功能,会对代码结构造成破环。如果想对函数增加新的功能,并且不修改原有函数,且调用方式不做出改变的话,就要使用装饰器。
简单的装饰器实现:
def limit(func):
def addlimit():
print("use limit")
func()
return addlimit
def test():
print("this is a function")
test = limit(test)
test()
函数 limit
就是一个装饰器,它把函数 test
当作参数传入,在内层函数中增加功能后,又把内层函数返回,重新赋值给 test
变量。这里使用到了闭包,外层函数负责接收要修饰的函数,返回修饰后的函数,内层函数赋值修饰传入的函数。
装饰器的使用有一种简写方式,就是在函数定义之前使用 @
+装饰器名字:
def limit(func):
def addlimit():
print("use limit")
func()
return addlimit
@limit # 相当于test = limit(test)
def test():
print("this is a function")
test()
@limit
便相当于 test = limit(test)
。
使用@语法糖 便相当于把定义在后边的函数当作参数传入装饰器。
def limit(func):
def addlimit(a, b):
print("use limit")
func(a, b)
return addlimit
@limit
def test(a, b):
print("I tell you : %s" % a)
print("I tell you : %s" % b)
a = "hello"
b = "hi"
test(a, b)
因为装饰器调用的是闭包中的内部函数,所以我们先在内部函数接收参数,再传递给被装饰的函数。这样,参数经过传递便被传递给了原有的 test
函数。
但是,如果其他函数不是两个参数,在使用该装饰器的时候,便会运行错误,为了装饰器的通用性,我们可以用不定长位置参数 *args
和关键字参数 **kwargs
def limit(func):
def addlimit(*args, **kwargs):
print("use limit")
func(*args, **kwargs)
return addlimit
@limit
def test(a, b):
print("I tell you : %s" % a)
print("I tell you : %s" % b)
@limit
def test1(a, b, c):
print("I tell you : %s" % a)
print("I tell you : %s" % b)
print("I tell you : %s" % c)
a = "hello"
b = "hi"
test(a, b)
print()
test1(a, b, c = "hello world")
运行结果:
use limit
I tell you : hello
I tell you : hi
use limit
I tell you : hello
I tell you : hi
I tell you : hello world
def limit(func):
def addlimit():
print("use limit")
return func()
return addlimit
@limit
def test():
print("this is a function")
return "I tell you: hello"
print(test())
直接在闭包内部函数返回原函数的调用结果即可。
def limit(func):
def addlimit(*args, **kwargs):
print("use limit")
return func(*args, **kwargs)
return addlimit
三层装饰器可以在原有装饰器的基础上,设置额外的外部变量。
def limit_arg(arg):
def limit(func):
def addlimit():
print("use limit--%s" % arg)
func()
return addlimit
return limit
@limit_arg("hello")
def test():
print("this is a function")
test()
运行结果:
use limit--hello
this is a function
test()
相当于:
limit_arg("hello")(test)()
装饰器不仅是一个函数,还可以是一个类。
class Limit(object):
def __init__(self, func):
self.func = func
def __call__(self):
print("use limit")
self.func()
@Limit
def test():
print("this is a function")
test()
魔术方法 __call__
方法可以让类的实例对象像函数一样被调用。
类装饰器对比函数装饰器具有灵活度大、高内聚、封装性等优点。
def add_a(func):
def a():
return "<a href=‘mxuanli.cn‘>" + func() + "</a>"
return a
def add_h1(func):
def h1():
return "<h1>" + func() + "</h1>"
return h1
@add_a
@add_h1
def test():
return "hello world"
print(test())
运行结果:
<a href=‘mxuanli.cn‘><h1>hello world</h1></a>
标签:通用 灵活 操作 %s 接收 函数赋值 内聚 pre 结构
原文地址:https://www.cnblogs.com/mxuanli/p/9804728.html