zoukankan      html  css  js  c++  java
  • [Python之路] 闭包

    一、思考一个问题

    我们要给定一个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)
  • 相关阅读:
    [置顶] kubernetes资源类型--DaemonSet
    Docker容器的自动化监控实现
    [置顶] docker--基础镜像和dockerfile
    Target runtime Apache Tomcat v8.0 is not defined
    jeecg-org.jeecgframework.web.system.listener.InitListener
    tomcat:利用tomcat部署war包格式的项目
    更换jdk版本:jdk1.8更换为jdk1.7之后输入java -version还是出现1.8的版本号
    Maven项目下WEB-INFO目录下没有编译的classes文件
    解决maven web项目Cannot detect Web Project version. Please specify version of Web Project through...的错误
    如何提高maven的下载速度:享受一下mvn时飞的感觉
  • 原文地址:https://www.cnblogs.com/leokale-zz/p/11966903.html
Copyright © 2011-2022 走看看