python闭包的实例详解

下面我将为您详细讲解 "Python闭包的实例详解"。

什么是闭包

在介绍闭包的实例之前,我们先来了解闭包的概念。闭包是指在函数内部再定义函数,并且这个内部函数可以引用到外部函数中的变量,叫做闭包。

下面是一个简单的例子,函数 counter 返回了内部函数 add,而 add 引用了外层的变量 n

def counter(n):
    def add(m):
        return m + n
    return add

闭包的基本特点

  1. 闭包函数可以访问外部函数中定义的变量,这些变量只在外部函数的作用域中才能访问;
  2. 当外部函数被调用时,外部函数中的变量被保存在内存中,内部函数可以随时访问这些变量,并且即使在外部函数执行完之后,这些变量的值也不会被释放掉;
  3. 内部函数不能改变外部函数中定义的变量,它只能进行读取操作。

闭包的实例

实例一:计数器

def make_counter():
    count = 0
    def counter():
        nonlocal count
        count += 1
        return count
    return counter

这个例子展示了如何使用闭包实现计数器。make_counter 函数返回了内部函数 counter,而 counter 函数每次被调用时都会自增 count 变量的值。

我们可以使用以下方式来使用 make_counter 函数创建计数器:

counter = make_counter()
print(counter()) # 输出 1
print(counter()) # 输出 2
print(counter()) # 输出 3

实例二:循环绑定事件

def bind_events():
    events = []
    for i in range(3):
        events.append(lambda: print(i))
    return events

这个例子展示了闭包的一个潜在问题。函数 bind_events 返回了一个包含了三个函数的列表 events,这三个函数都是对一个变量 i 的引用。

我们可以使用以下方式来测试 bind_events 函数:

for event in bind_events():
    event()

预期的输出应该是:

0
1
2

但实际上,输出的却是:

2
2
2

这是因为当执行这三个函数时,它们所引用的 i 已经变成了循环结束时的值,即 i=2。要解决这个问题,我们可以使用默认参数来保存 i 的值:

def bind_events():
    events = []
    for i in range(3):
        events.append(lambda i=i: print(i))
    return events

现在,我们再执行下面的代码,输出就变成了预期的结果:

for event in bind_events():
    event()

输出:

0
1
2

所以,闭包虽然可以方便地共享变量,但是也需要小心使用,避免出现上述问题。

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:python闭包的实例详解 - Python技术站

(0)
上一篇 2023年6月10日
下一篇 2023年6月10日

相关文章

  • js立即执行函数: (function ( ){})( ) 与 (function ( ){}( )) 有什么区别?

    JavaScript立即执行函数是一种特殊的函数表达式,它可以立即自动运行,而不需要在其他地方调用。在JavaScript中,我们可以使用两种不同的声明方法来创建立即执行函数。第一种是使用普通的函数声明方法,代码如下: (function() { //code to be executed immediately })(); 这种声明方法使用一个内部或匿名函…

    JavaScript 2023年5月27日
    00
  • DOM基础教程之使用DOM + Css

    DOM(Document Object Model)是一种用于处理HTML,XML等文档的接口。使用DOM结合CSS可以实现丰富多彩的网页效果,接下来我们来讲解使用DOM和CSS的完整攻略。 步骤1:在HTML中引入CSS文件 首先,在HTML头部引入CSS文件,以便在DOM中使用CSS样式。 <head> <link rel="…

    JavaScript 2023年6月10日
    00
  • JS 仿Flash动画放大/缩小容器

    下面我将为你详细讲解“JS 仿Flash动画放大/缩小容器”的完整攻略。 攻略概述 这个攻略解决的问题是实现JS仿Flash的动画效果,主要通过控制容器的大小和位置来实现缩放和移动的效果,同时也可以在动画播放过程中改变容器中的内容。具体实现过程分为以下几个步骤: 创建HTML和CSS代码,用来定义容器和样式。 通过JavaScript获取容器对象,并设置其初…

    JavaScript 2023年6月10日
    00
  • JavaScript实现大文件上传的示例代码

    让我为您详细讲解JavaScript实现大文件上传的完整攻略。 1. 概述 在网页上传大文件时,为了提高用户体验,通常会采用分片上传的方式,即将大文件分为若干个小块逐个上传,上传成功之后再将这些小块合并成一个完整的文件。而在实现分片上传时,JavaScript是必不可少的。 2. 实现步骤 下面是一份JavaScript实现大文件上传的示例代码,包括以下步骤…

    JavaScript 2023年5月27日
    00
  • 一起来学习一下JavaScript的事件流

    关于JavaScript事件流,我为大家准备了一份完整攻略,一起来学习一下。 什么是JavaScript事件流 JavaScript事件流是指浏览器中发生事件(如鼠标点击、键盘输入等)时,事件在DOM树结构中按照特定顺序发送和处理的过程。这个过程包含三个阶段:捕获阶段、目标阶段和冒泡阶段。 捕获阶段 在事件到达目标元素之前,从根节点到目标元素之间的所有节点都…

    JavaScript 2023年6月10日
    00
  • javascript使用正则控制input输入框允许输入的值方法大全

    JavaScript使用正则控制input输入框允许输入的值方法大全 在开发前端网页时,有时我们需要对输入框的输入内容进行限制,只允许输入特定类型的数据,这时就可以使用JavaScript的正则表达式来控制。 以下列出了几种常见的限制方式和相应的正则表达式: 限制只允许输入数字 <input type="text" onkeyup=…

    JavaScript 2023年6月11日
    00
  • 使用JavaScript通过前端发送电子邮件

    下面是“使用JavaScript通过前端发送电子邮件”的完整攻略: 1.准备工作 要通过前端使用JavaScript发送电子邮件,你需要使用邮件服务提供商的API,本文以SendGrid为例进行说明。在使用SendGrid之前,你需要完成以下准备工作: 注册SendGrid账户并创建API密钥 创建一个用于发送邮件的HTML表单 2.获取API密钥 在Sen…

    JavaScript 2023年6月10日
    00
  • document.execCommand()的用法小结

    标题:document.execCommand()的用法小结 简介 document.execCommand() 是一个可以对富文本编辑器进行操作的 JavaScript API。它可用于设置文本样式、格式化或插入内容等操作。它最初在 Internet Explorer 5.5 中引入,但现在大多数主流浏览器都支持这个 API 了。 语法 document.…

    JavaScript 2023年6月11日
    00
合作推广
合作推广
分享本页
返回顶部