python的闭包

liusarazhang 2019-12-01

一、思考一个问题

我们要给定一个x,要求一条直线上x对应的y的值。公式是y = kx+b。

我们需要用k,b来确定这条直线,则我们实现的函数应该有3个参数:

def line(k, b, x):
    print(k * x + b)


line(1, 3, 4)
line(1, 3, 5)
line(1, 3, 6)

可以看到,我们每次修改x都要重新传入k和b。

我们也可以用全局变量来实现:

k = 1
b = 3


def line(x):
    print(k * x + b)


line(4)
line(5)
line(6)

k和b为全局变量,但如果我们想要另外一条不同的直线时,则还需要重新定义k和b的值,同样不合理。而且全局变量会暴露给其他不相关的函数,容易造成冲突,或代码混乱。

用面向对象来实现:

class Line(object):
    def __init__(self, k, b):
        self.k = k
        self.b = b

    def create_y(self, x):
        print(self.k * x + self.b)


l1 = Line(1, 3)
l1.create_y(4)
l1.create_y(5)
l1.create_y(6)

用类和对象来实现肯定是可以的,但是这么一个简单的功能使用面向对象比较浪费资源。

二、使用闭包

def line(k, b):
    def create_y(x):
        print(k * x + b)

    return create_y


l1 = line(1, 3)
l1(4)
l1(5)
l1(6)

从以上代码可以直观的看到,闭包有以下几个条件:

1.在一个外函数中定义了一个内函数。

2.内函数里运用了外函数的临时变量。

3.并且外函数的返回值是内函数的引用。

一个函数结束的时候,会把自己的临时变量都释放给内存,之后变量都不存在了。一般情况下,确实是这样的。但是闭包是一个特别的情况。外部函数发现,自己的临时变量会在将来的内部函数中用到,自己在结束的时候,返回内函数的同时,会把外函数的临时变量和内函数绑定在一起。所以外函数已经结束了,调用内函数的时候仍然能够使用外函数的临时变量。

三、内层函数修改外层函数临时变量

def line(k, b):
    multi = 10;

    def create_y(x):
        print((k * x + b) * multi)

    return create_y


l1 = line(1, 3)
l1(4)
l1(5)
l1(6)

假设外层函数中有一个multi变量,这个变量可以提供给内层函数访问,但是如果内层函数想修改这个multi变量怎么办呢?

在python3.x中,我们可以使用nonlocal关键字来实现:

def line(k, b):
    multi = 10;

    def create_y(x):
        nonlocal multi
        multi = 5
        print((k * x + b) * multi)

    return create_y


l1 = line(1, 3)
l1(4)
l1(5)
l1(6)

这个nonlocal类似与global的作用,但是global是用于修改全局变量,而nonlocal是用于修改闭包中的外层临时变量。

如果在python2.x中,则不存在nonlocal关键字,我们可以通过将multi变为可变类型数据来实现:

def line(k, b):
    # python2.x中将multi变为列表
    multi = [10]

    def create_y(x):
        # 可以对外层函数的列表进行修改
        multi[0] = 5
        print((k * x + b) * multi[0])

    return create_y


l1 = line(1, 3)
l1(4)
l1(5)
l1(6)

四、闭包和函数、对象等的区别

函数(包含匿名函数):只是功能代码,不包含数据。

对象:包含数据和功能实现。

闭包:包含数据和功能实现。数据指外层函数接收到的参数以及他的局部变量,功能指内层函数的功能。

当函数、匿名函数、对象和闭包做为实参传递时,他们有什么区别????

1.函数和匿名函数被当做实参传递时,传递的是功能的引用,可以通过该引用调用他们实现的功能。但数据需要另外提供。

2.闭包被当做实参传递时,其实传递了数据+功能。例如:

def line(k, b):
    # python2.x中将multi变为列表
    multi = [10]

    def create_y(x):
        # 可以对外层函数的列表进行修改
        multi[0] = 5
        print((k * x + b) * multi[0])

    return create_y


def function(func):
    func(5)


function(line(1, 3))

我们可以看到,闭包line被传入function,实际上带着数据k=1,b=3。

而如果是普通函数:

def line_norm(k, b, x):
    print(k * x + b)


def function(func):
    func(1, 3, 5)


function(line_norm)

这里的k=1,b=3是function函数自己提供的。

3.对象被当做实参传递时,该对象中的成员属性作为数据,成员方法作为方法,都被传递给function函数。

class Line(object):
    def __init__(self, k, b):
        self.k = k
        self.b = b

    def create_y(self, x):
        print(self.k * x + self.b)


line1 = Line(1, 3)


def function(inst):
    inst.create_y(5)


function(line1)

相关推荐