表妹又来问JavaScript的闭包啦,这就给你把来龙去脉讲清楚

Posted Ultraman_agul

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了表妹又来问JavaScript的闭包啦,这就给你把来龙去脉讲清楚相关的知识,希望对你有一定的参考价值。

什么是闭包

闭包是指有权访问另一个函数作用域中的变量的函数。

函数中的name,test1不会被垃圾回收机制回收,直到页面关闭

function myName() {
    let test1 = 1
    let test2 = 2
    let name = 'agul'
    let foo = {
        getName: function () {
            return name
        },
        setName: function (value) {
            name = value
            console.log(test1)
        }
    }
    return foo
}

let bar = myName()
bar.setName('pjh')
console.log(bar.getName())

1. 闭包形成的原理

  • 作用域链,当前作用域可以访问上级作用域中的变量
  • 全局变量只用页面关闭才会销毁

2. 闭包解决的问题

  • 函数作用域中的变量在函数执行结束就会销毁,但是有时候我们并不希望变量销毁
  • 在函数外部可以访问函数内部的变量

3 闭包带来的问题

  • 容易造成内存泄露

    • 内存泄漏:占用的内存没有及时释放,内存泄露积累多了就容易导致内存溢出

      • 闭包

        function fn1() {
            var a = 4
            function fn2() {
              console.log(++a)
            }
            return fn2
          }
          var f = fn1()
          f()
        

4 闭包的应用

  • 4.1模仿块级作用域

for (var i = 0; i < 5; i++) {
    (function (j) {
        setTimeout(() => {
            console.log(j)
        }, j * 1000)
    })(i)
}
//相当于
for (let i = 0; i < 5; i++) {
    setTimeout(() => {
        console.log(i)
    }, i * 1000)
}
  • 4.2埋点计数器

两个按钮,分别记录点击次数

<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
</head>

<body>
    <button>test1</button>
    <button>test2</button>
    <script>
        function count() {
            var num = 0
            return function () {
                return ++num
            }
        }
        var fn1 = count()
        var fn2 = count()
        document.querySelectorAll('button')[0].onclick = function () { console.log(fn1()) }
        document.querySelectorAll('button')[1].onclick = function () { console.log(fn2()) }
    </script>
</body>

</html>
  • 4.2柯里化

传入一个参数返回一个函数,返回的函数又能够传入参数使用。

   function curryingCheck(reg) {
      return function (txt) {
           return reg.test(txt)
       }
   }
  
   var isEmail = curryingCheck(/^\\w+([-+.]\\w+)*@\\w+([-.]\\w+)*\\.\\w+([-.]\\w+)*$/)
      console.log(isEmail('wyn@nowcoder.com'))      // false

上方代码是传入验证邮箱的正则调用函数,向返回函数中传入验证的值就能够进行验证真假了。验证其他也是可以如此使用~

以上是关于表妹又来问JavaScript的闭包啦,这就给你把来龙去脉讲清楚的主要内容,如果未能解决你的问题,请参考以下文章

如何用函数式编程思想优化业务代码,这就给你安排上!

javascript详解javascript闭包 — 大家准备好瓜子,我要开始讲故事啦~~

JavaScript 经典之一 闭包

javascript 执行环境,作用域链和闭包

javascript解决闭包漏洞的一个问题

javascript解决闭包漏洞的一个问题