简单了解JavaScript闭包

本篇文章给大家带来了关于javascript的相关知识,其中主要介绍了关于javascript闭包的相关问题,闭包的概念有很多版本,不同的地方对闭包的说法不一,下面一起来看一下,希望对大家有帮助。

简单了解JavaScript闭包

什么是闭包?

闭包的概念是有很多版本,不同的地方对闭包的说法不一

维基百科:在计算机科学中,闭包(英语:Closure),又称词法闭包(Lexical Closure)或函数闭包(function closures),是在支持头等函数的编程语言中实现词法绑定的一种技术。

MDN: 闭包(closure)是一个函数以及其捆绑的周边环境状态(lexical environment词法环境)的引用的组合。

个人理解:

立即学习“Java免费学习笔记(深入)”;

闭包是一个函数(返回一个函数)返回的函数保存了对外变量引用

一个简单的示例

function fn() {    let num = 1;    return function (n) {        return n + num    }}let rFn = fn()let newN = rFn(3) // 4

num 变量作用域在 fn 函数中, rFn 函数却能访问 num 变量,这就是闭包函数能访问外部函数变量。

从浏览器调试和 VSCode Nodejs 调试看闭包

浏览器

image.png

VS Code 配合 Node.js

image.png

看到 Closure 中 fn 是闭包函数,其中保存 num 变量。

一个经典的闭包:单线程事件机制+循环问题,以及解决办法

for (var i = 1; i  {    console.log(i);  }, i * 1000);}

输出的结果都是 6,为什么?

for 循环是同步任务setTimeout 异步任务

for 循环一次,就会将 setTimeout 异步任务加入到浏览器的异步任务队列中,同步任务完成之后,再从异步任务中拿新任务在线程中执行。由于 setTimeout 能够访问外部变量 i, 当同步任务完成之后,i 已经变成了6, setTimeout 中能够访问变量 i 都是 6。

豆包爱学 豆包爱学

豆包旗下AI学习应用

豆包爱学 674 查看详情 豆包爱学

解决办法1:使用 let 声明

for (var i = 1; i  {    console.log(i);  }, i * 1000);}

解决办法2:自执行函数 + 闭包

for (var i = 1; i  {    console.log(i);  }, i * 1000)  })(i)}

解决办法3:setTimeout 传递第三参数

第三个参数意思:附加参数,一旦定时器到期,它们会作为参数传递给要执行的函数

for (var i = 1; i  {    console.log(j);  }, 1000 * i, i);}

闭包与函数科里化

function add(num) {  return function (y) {    return num + y;  };};let incOneFn = add(1); let n = incOneFn(1);  // 2let decOneFn = add(-1); let m = decOneFn(1); // 0

add 函数的参数保存了闭包函数变量。

实际作用

在函数式编程闭包有非常重要的作用,lodash 等早期工具函数弥补 javascript 缺陷的工具函数,有大量的闭包的使用场景。

使用场景

创建私有变量延长变量生命周期

节流函数

防止滚动行为,过度执行函数,必须要节流, 节流函数接受 函数 + 时间作为参数,都是闭包中变量,以下是一个简单 setTimeout 版本:

function throttle(fn, time=300){    var t = null;    return function(){        if(t) return;        t = setTimeout(() => {            fn.call(this);            t = null;        }, time);    }}

防抖函数

一个简单的基于 setTimeout 防抖的函数的实现

function debounce(fn,wait){    var timer = null;    return function(){        if(timer !== null){            clearTimeout(timer);        }        timer = setTimeout(fn,wait);    }}

React.useCallback 闭包陷阱问题

问题说明:父/子 组件关系, 父子组件都能使用 click 事件同时修改 state 数据, 并且子组件拿到传递下的 props 事件属性,是经过 useCallback 优化过的。也就是这个被优化过的函数,存在闭包陷阱,(保存一直是初始 state 值)

import { useState, useCallback, memo } from "react";const ChildWithMemo = memo((props: any) => {  return (    
);});const Parent = () => { const [count, setCount] = useState(1); const handleClickWithUseCallback = useCallback(() => { console.log(count); }, []); // 注意这里是不能监听 count, 因为每次变化都会重新绑定,造成造成子组件重新渲染 return (
parent count : {count}
);};export default Parent

ChildWithMemo 使用 memo 进行优化,handleClickWithUseCallback 使用 useCallback 优化

问题是点击子组件时候,输出的 count 是初始值(被闭包了)。

解决办法就是使用 useRef 保存操作变量函数:

import { useState, useCallback, memo, useRef } from "react";const ChildWithMemo = memo((props: any) => {  console.log("rendered children")  return (    
);});const Parent = () => { const [count, setCount] = useState(1); const countRef = useRef(null) countRef.current = () => { console.log(count); } return (
parent count : {count}
);};export default Parent

针对这个问题,React 曾经认可过社区提出的增加 useEvent 方案,但是后面 useEvent 语义问题被废弃了,对于渲染优化 React 采用了编译优化的方案。其实类似的问题也会发生在 useEffect 中,使用时要注意闭包陷阱。

性能问题

闭包不要随意定义,定义了一定找到合适的位置进行销毁。因为闭包的变量保存在内存中,不会被销毁,占用较高的内存。

使用 chrome 面板功能 timeline + profiles 面板

打开开发者工具,选择 Timeline 面板在顶部的Capture字段里面勾选 Memory点击左上角的录制按钮。在页面上进行各种操作,模拟用户的使用情况。一段时间后,点击对话框的 stop 按钮,面板上就会显示这段时间的内存占用情况。

【相关推荐:JavaScript视频教程、web前端】

以上就是简单了解JavaScript闭包的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/544617.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月9日 16:48:59
下一篇 2025年11月9日 16:50:11

相关推荐

  • Go语言中条件式变量赋值的最佳实践

    本文探讨了如何在go语言中实现类似于javascript中利用逻辑或运算符进行条件式变量赋值的模式。go语言由于其强类型特性和缺乏非布尔类型的“真值/假值”概念,以及没有三元运算符,需要采用更显式的方法,即通过`if/else if/else`结构来安全、清晰地完成变量的条件式赋值。 在JavaSc…

    2025年12月16日
    000
  • Go语言中实现多选项变量赋值的惯用方法

    本文探讨如何在go语言中实现类似javascript `var foo = a || b || c;` 的多选项变量赋值逻辑。go语言强调类型安全,不支持字符串等非布尔类型的“真值”判断,也未提供三元运算符。因此,应采用清晰的`if-else if-else`结构进行显式条件判断,确保代码的可读性和…

    2025年12月16日
    000
  • 解决Revel框架静态文件加载异常:GOPATH与文件路径疑难解析

    本教程旨在解决revel框架中静态文件加载异常(如显示旧版本或不完整文件)的问题。核心原因常与`gopath`配置不当或文件路径解析错误有关。文章将提供详细的排查步骤,包括`gopath`一致性检查、文件副本识别以及利用系统工具追踪文件访问路径,并强调在`gopath`内进行revel应用开发的最佳…

    2025年12月16日
    000
  • Go语言中从多个选项声明变量的惯用方法

    本文探讨了go语言中如何实现类似javascript的从多个备选值中声明变量的模式。go语言强调类型安全和代码显式性,不支持javascript中字符串的“真值”概念,也没有三元运算符。因此,在go中,应使用清晰的`if-else if-else`结构来检查变量的有效性并进行赋值,以确保代码的健壮性…

    2025年12月16日
    000
  • Go语言包函数导出机制:通过命名约定实现可见性控制

    go语言的包函数导出机制简洁而独特,它并非依赖于特定的关键字或模块对象,而是通过标识符的首字母大小写来控制。当函数名以大写字母开头时,该函数将从包中导出,可在包外部被访问;而以小写字母开头的函数则仅限于包内部使用。这一设计原则同样适用于变量、类型和结构体字段。 在Go语言中,对于初次接触的开发者,尤…

    2025年12月16日
    000
  • Go Gorilla Sessions:会话管理与变量设置实践指南

    本文详细介绍了如何在go语言的gorilla web框架中使用`gorilla/sessions`包进行会话管理。内容涵盖了会话存储的初始化、会话的获取与创建、会话变量的设置与读取,以及关键的会话保存操作,并提供了完整的代码示例和重要注意事项,旨在帮助开发者高效、安全地实现基于cookie的会话机制…

    2025年12月16日
    000
  • Go Web服务器中Goroutine误用:文件服务空白页的根源与高效优化实践

    本文深入剖析了在go web服务器中,将文件加载函数作为goroutine调用导致响应空白页的常见问题。核心原因在于http处理器在goroutine完成写入前过早返回,导致空响应被发送。文章详细解释了go http请求处理机制,揭示了`ioutil.readfile`的性能瓶颈,并提供了基于`os…

    2025年12月16日
    000
  • 深入理解Go HTTP服务器与Goroutine:避免常见陷阱与优化文件服务

    在go http服务器中,直接将页面加载逻辑封装为goroutine可能导致空白响应,因为http处理器期望同步完成请求。本文将深入探讨go http处理器的生命周期,解释为何不当使用goroutine会中断响应流,并提供使用`os.open`与`io.copy`优化文件流式传输的方法,同时推荐`h…

    2025年12月16日
    000
  • Go Gorilla 框架会话管理:深度解析与实践指南

    本文旨在提供一份关于如何在 go 语言中使用 gorilla sessions 框架进行会话管理的全面教程。我们将详细探讨会话存储的初始化、会话的获取与设置、关键的 cookie 选项配置,以及确保会话数据正确保存到客户端浏览器的核心步骤,帮助开发者高效、安全地实现用户会话功能。 正文 1. 引言 …

    2025年12月16日
    000
  • Go Gorilla Sessions:深入理解与实践会话管理

    本教程详细讲解了如何在go语言中使用gorilla sessions框架进行会话管理。内容涵盖cookiestore的初始化、会话的获取与创建、会话值的设置与持久化,以及会话选项的配置,旨在帮助开发者构建安全、可靠的web应用会话机制。 1. 引言:Go Gorilla Sessions 简介 在W…

    2025年12月16日
    000
  • 自定义日志处理与用户行为分析:从文件系统到专业工具的最佳实践

    本教程探讨了自定义日志格式的解析、存储与分析策略。针对用户行为日志,文章指出传统文件系统存储的局限性,并推荐转向事件驱动的专业分析平台,如Mixpanel或Keen.io,以实现高效数据洞察与可视化。同时,也讨论了Unix工具、编程语言在日志解析中的应用场景,强调了可视化在理解数据中的核心作用。 在…

    2025年12月16日
    000
  • 如何在Go语言中使用Gorilla Sessions框架管理HTTP会话

    本教程全面指导如何在Go应用程序中利用Gorilla Sessions框架实现和管理HTTP会话。内容涵盖CookieStore的设置、会话的初始化与检索、会话值的设置与持久化,以及安全且健壮的会话选项配置,确保HTTP Cookie的正确处理。 1. Gorilla Sessions简介 HTTP…

    2025年12月16日
    000
  • 优化日志处理:从文件系统到事件分析的转变

    本文探讨了高效处理日志以理解用户行为的方法。针对传统文件系统日志存储在行为分析方面的局限性,我们推荐采用事件驱动的分析平台,如mixpanel或keen.io。这些平台通过结构化事件收集和强大的可视化功能,能够更直接、更深入地洞察用户行为,避免了手动解析和关联大量原始日志的复杂性。 在现代应用开发中…

    2025年12月16日
    000
  • 日志处理与用户行为分析:从传统解析到现代事件驱动方法

    本文深入探讨了自定义日志格式的解析与用户行为分析策略。针对传统文件系统组织日志的局限性,我们提出了一种现代的事件驱动方法。通过利用专业的事件分析平台和可视化%ignore_a_1%,可以更高效地收集、分析用户行为数据,并从中提取有价值的洞察,从而超越单纯的日志存储,实现数据驱动的决策。 在复杂的应用…

    2025年12月16日
    000
  • Golang html/template安全生成HTML示例

    使用 html/template 可自动转义变量防止 XSS,如 {{.Username}} 会转义恶意脚本;需插入可信 HTML 时可使用 template.HTML 类型,但必须确保内容安全;应避免手动拼接 HTML,而将原始数据交由模板处理,以保证各上下文正确转义。 Go 的 html/tem…

    2025年12月16日
    000
  • 在Go语言Web服务前置Nginx的优势

    在go语言web服务前置nginx作为反向代理,能带来显著的性能、安全和管理效益。nginx擅长处理日志记录、ssl/tls终止、http/2支持、内容压缩、http头部管理以及高效静态资源服务等“web服务器”任务,从而让go应用专注于业务逻辑,避免重复造轮子,构建更健壮、可扩展的系统。 将Ngi…

    2025年12月16日
    000
  • Go语言中获取变量类型字符串的实用方法

    在go语言中,获取变量的类型并以字符串形式打印是一个常见需求。本文将介绍如何使用`fmt.printf`函数的`%t`格式化动词来高效、简洁地实现这一目标,避免了类似javascript `typeof`或python `type`操作符的误区。通过一个简单的示例,读者将掌握在go中获取变量类型字符…

    2025年12月16日
    000
  • Golang如何开发在线计算器项目

    用Golang开发在线计算器需前后端协作:前端HTML页面通过fetch发送表达式,后端Go程序用net/http处理POST请求,借助govaluate解析计算并返回JSON结果,主函数注册/calculate路由和静态文件服务,项目结构清晰,可快速搭建运行。 用Golang开发一个在线计算器项目…

    2025年12月16日
    000
  • Golang如何构建简单的博客评论系统

    先定义评论结构体,包含ID、作者、内容和创建时间。使用切片和互斥锁在内存中存储评论,保证并发安全。通过net/http实现GET /comments获取所有评论,POST /comment提交新评论,处理JSON数据并校验字段。前端可嵌入HTML表单,用JavaScript调用API实现交互。核心是…

    2025年12月16日
    000
  • Go语言:获取变量类型字符串的两种实用方法

    本文深入探讨了go语言中获取变量类型并以字符串形式输出的两种主要方法。首先介绍如何利用`fmt.printf`函数的`%t`格式化动词进行快速打印,适用于调试和日志记录。随后,详细阐述了如何借助`reflect`包的`typeof`函数来编程获取变量的类型字符串,包括`string()`和`name…

    2025年12月16日
    000

发表回复

登录后才能评论
关注微信