我的编程空间,编程开发者的网络收藏夹
学习永远不晚

JavaScript闭包实例代码分析

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

JavaScript闭包实例代码分析

这篇文章主要介绍了JavaScript闭包实例代码分析的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇JavaScript闭包实例代码分析文章都会有所收获,下面我们一起来看看吧。

什么是闭包?

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

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

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

个人理解:

  • 闭包是一个函数(返回一个函数)

  • 返回的函数保存了对外变量引用

一个简单的示例

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 调试看闭包

  • 浏览器

JavaScript闭包实例代码分析

  • VS Code 配合 Node.js

JavaScript闭包实例代码分析

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

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

for (var i = 1; i <= 5; i++) {  setTimeout(() => {    console.log(i);  }, i * 1000);}

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

  • for 循环是同步任务

  • setTimeout 异步任务

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

解决办法1:使用 let 声明

for (var i = 1; i <= 5; i++) {  setTimeout(() => {    console.log(i);  }, i * 1000);}

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

for (var i = 1; i <= 5; i++) {  (function(i){      setTimeout(() => {    console.log(i);  }, i * 1000)  })(i)}

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

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

for (var i = 1; i <= 5; i++) {  setTimeout((j) => {    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 (    <div>      <button onClick={props.handleClick}>Child click</button>    </div>  );});const Parent = () => {  const [count, setCount] = useState(1);  const handleClickWithUseCallback = useCallback(() => {    console.log(count);  }, []); // 注意这里是不能监听 count, 因为每次变化都会重新绑定,造成造成子组件重新渲染  return (    <div>      <div>parent count : {count}</div>      <button onClick={() => setCount(count + 1)}>click</button>      <ChildWithMemo handleClick={handleClickWithUseCallback} />    </div>  );};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 (    <div>      <button onClick={() => props.countRef.current()}>Child click</button>    </div>  );});const Parent = () => {  const [count, setCount] = useState(1);  const countRef = useRef<any>(null)  countRef.current = () => {    console.log(count);  }  return (    <div>      <div>parent count : {count}</div>      <button onClick={() => setCount(count + 1)}>click</button>      <ChildWithMemo countRef={countRef} />    </div>  );};export default Parent

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

性能问题

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

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

  1. 打开开发者工具,选择 Timeline 面板

  2. 在顶部的Capture字段里面勾选 Memory

  3. 点击左上角的录制按钮。

  4. 在页面上进行各种操作,模拟用户的使用情况。

  5. 一段时间后,点击对话框的 stop 按钮,面板上就会显示这段时间的内存占用情况。

关于“JavaScript闭包实例代码分析”这篇文章的内容就介绍到这里,感谢各位的阅读!相信大家对“JavaScript闭包实例代码分析”知识都有一定的了解,大家如果还想学习更多知识,欢迎关注编程网行业资讯频道。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

JavaScript闭包实例代码分析

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

JavaScript闭包实例代码分析

这篇文章主要介绍了JavaScript闭包实例代码分析的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇JavaScript闭包实例代码分析文章都会有所收获,下面我们一起来看看吧。什么是闭包?闭包的概念是有很多版本
2023-07-05

javascript中闭包closure的代码案例

这篇文章主要讲解了“javascript中闭包closure的代码案例”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“javascript中闭包closure的代码案例”吧!简介闭包closu
2023-06-06

Laravel枚举包standards实例代码分析

这篇文章主要讲解了“Laravel枚举包standards实例代码分析”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Laravel枚举包standards实例代码分析”吧!PrinsFran
2023-07-04

Java/JavaScript/ABAP代码重构实例分析

本篇内容介绍了“Java/JavaScript/ABAP代码重构实例分析”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!在方法里引入一个布尔类
2023-06-05

C++开放封闭原则实例代码分析

这篇文章主要介绍“C++开放封闭原则实例代码分析”,在日常操作中,相信很多人在C++开放封闭原则实例代码分析问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”C++开放封闭原则实例代码分析”的疑惑有所帮助!接下来
2023-07-05

JavaScript闭包原理及作用的示例分析

小编给大家分享一下JavaScript闭包原理及作用的示例分析,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!简介说明本文介绍JavaScript的闭包的作用、用途及其原理。闭包的定义闭包是指内部函数总是可以访问其所在的外部
2023-06-22

JavaScript代码优化的技巧实例分析

本篇内容主要讲解“JavaScript代码优化的技巧实例分析”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“JavaScript代码优化的技巧实例分析”吧!写在前面想要做到JavaScript的代
2023-07-02

JavaScript单行代码示例分析

这篇文章主要介绍JavaScript单行代码示例分析,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!什么是单行代码?单行代码是一种代码实践,其中我们仅用一行代码执行某些功能。01-随机获取布尔值此函数将使用Math.r
2023-06-15

Sendable 和 @Sendable 闭包代码实例详解

Sendable协议和闭包表明那些传递的值的公共API是否线程安全的向编译器传递了值。当没有公共修改器、有内部锁定系统或修改器实现了与值类型一样的复制写入时,公共API可以安全地跨并发域使用。
SendableSwift2024-12-01

Go语言中的闭包实例分析

这篇文章主要介绍“Go语言中的闭包实例分析”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Go语言中的闭包实例分析”文章能帮助大家解决问题。一、函数的变量作用域和可见性1.全局变量在main函数执行之
2023-07-02

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录