javascript闭包_JavaScript闭包教程–带有JS闭包示例代码

javascript闭包

Closures – many of you JavaScript devs have probably heard this term before. When I started my journey with JavaScript, I encountered closures often. And I think they're one of the most important and interesting concepts in JavaScript.

闭包–您中的许多JavaScript开发人员以前可能都听说过这个词。 当我开始使用JavaScript时,经常会遇到关闭。 而且我认为它们是JavaScript中最重要和最有趣的概念之一。

You don't think they're interesting? This often happens when you don’t understand a concept – you don’t find it interesting. (I don’t know if this happens to you or not, but this is the case with me).

你不觉得他们有趣吗? 当您不了解某个概念时,通常会发生这种情况–您不觉得它很有趣。 (我不知道这是否发生在您身上,但我就是这种情况)。

So in this article, I will try to make closures interesting for you.

因此,在本文中,我将尝试使闭包变得有趣。

Before going into the world of closures, let’s first understand lexical scoping. If you already know about it, skip the next part. Otherwise jump into it to better understand closures.

在进入闭包世界之前,让我们首先了解词法作用域 。 如果您已经知道它,请跳过下一部分。 否则,请跳入其中以更好地了解闭包。

词汇范围 (Lexical Scoping )

You may be thinking – I know local and global scope, but what the heck is lexical scope? I reacted the same way when I heard this term. Not to worry! Let’s take a closer look.

您可能在想–我知道本地和全球范围,但是词汇范围到底是什么呢? 听到这个学期时,我的React是一样的。 不要担心! 让我们仔细看看。

It’s simple like other two scopes:

就像其他两个作用域一样简单:

function greetCustomer() {var customerName = "anchal";function greetingMsg() {console.log("Hi! " + customerName); // Hi! anchal}greetingMsg();
}

You can see from the above output that the inner function can access the outer function's variable. This is lexical scoping, where the scope and value of a variable is determined by where it is defined/created (that is, its position in the code). Got it?

从上面的输出中可以看到,内部函数可以访问外部函数的变量。 这是词法作用域,其中变量的范围和值由定义/创建的位置(即其在代码中的位置)确定。 得到它了?

I know that last bit might have confused you. So let me take you deeper. Did you know that lexical scoping is also known as static scoping? Yes, that's its other name.

我知道最后一点可能会让您感到困惑。 因此,让我更深入地介绍您。 您是否知道词法作用域也称为静态作用域 ? 是的,这是它的别称。

There is also dynamic scoping, which some programming languages support. Why have I mentioned dynamic scoping? Because it can help you better understand lexical scoping.

还有动态范围 ,某些编程语言支持。 为什么提到动态范围? 因为它可以帮助您更好地理解词法作用域。

Let’s look at some examples:

让我们看一些例子:

function greetingMsg() {console.log(customerName);// ReferenceError: customerName is not defined
}function greetCustomer() {var customerName = "anchal";greetingMsg();
}greetCustomer();

Do you agree with the output? Yes, it will give a reference error. This is because both functions don’t have access to each other’s scope, as they are defined separately.

您是否同意输出? 是的,它将给出参考错误。 这是因为这两个函数无法访问彼此的范围,因为它们是分别定义的。

Let’s look at another example:

让我们看另一个例子:

function addNumbers(number1) {console.log(number1 + number2);
}function addNumbersGenerate() {var number2 = 10;addNumbers(number2);
}addNumbersGenerate();

The above output will be 20 for a dynamically scoped language. Languages that support lexical scoping will give referenceError: number2 is not defined. Why?

对于动态范围的语言,以上输出将为20。 支持词法作用域的语言会给 referenceError: number2 is not defined 。 为什么?

Because in dynamic scoping, searching takes place in the local function first, then it goes into the function that called that local function. Then it searches in the function that called that function, and so on, up the call stack.

因为在动态作用域中,搜索首先在本地函数中进行,然后才进入调用该本地函数的函数中。 然后,它在调用函数的函数中进行搜索,依此类推,直到调用堆栈。

Its name is self explanatory – “dynamic” means change. The scope and value of variable can be different as it depends on from where the function is called. The meaning of a variable can change at runtime.

它的名字不言自明–“动态”意味着变化。 变量的范围和值可以不同,这取决于调用函数的位置。 变量的含义可以在运行时更改。

Got the gist of dynamic scoping? If yes, then just remember that lexical scoping is its opposite.

有动态范围的要点吗? 如果是,那么请记住词汇范围是相反的。

In lexical scoping, searching takes place in the local function first, then it goes into the function inside which that function is defined. Then it searches in the function inside which that function is defined and so on.

在词法作用域中,搜索首先在局部函数中进行,然后进入定义函数的函数中。 然后,它在定义函数的函数中进行搜索,依此类推。

So, lexical or static scoping means the scope and value of a variable is determined from where it is defined. It doesn’t change.

因此, 词汇静态作用域意味着从定义变量的位置确定变量的范围和值。 它没有改变。

Let’s again look at the above example and try to figure out the output on your own. Just one twist – declare number2 at the top:

让我们再次看一下上面的示例,尝试自己找出输出。 只需一转-在顶部声明数字number2

var number2 = 2;
function addNumbers(number1) {console.log(number1 + number2);
}function addNumbersGenerate() {var number2 = 10;addNumbers(number2);
}addNumbersGenerate();

Do you know what the output will be?

您知道输出是什么吗?

Correct – it’s 12 for lexically scoped languages. This is because first, it looks into an addNumbers function (innermost scope) then it searches inwards, where this function is defined. As it gets the number2 variable, meaning the output is 12.

正确–词汇范围语言为12。 这是因为,首先,它查看一个addNumbers函数(最内部的作用域),然后向内搜索,在其中定义了此函数。 当它获得number2变量时,意味着输出为12。

You may be wondering why I have spent so much time on lexical scoping here. This is a closure article, not one about lexical scoping. But if you don’t know about lexical scoping then you will not understand closures.

您可能想知道为什么我在这里花了这么多时间进行词汇界定。 这是一篇闭篇文章,而不是有关词法作用域的文章。 但是,如果您不了解词法作用域,那么您将不会理解闭包。

Why? You will get your answer when we look at the definition of a closure. So let’s get into the track and get back to closures.

为什么? 当我们查看闭包的定义时,您会得到答案。 因此,让我们进入正轨,回到闭包。

什么是闭包? (What is a Closure? )

Let’s look at the definition of a closure:

让我们看一下闭包的定义:

Closure is created when an inner function has access to its outer function variables and arguments. The inner function has access to –

当内部函数可以访问其外部函数的变量和参数时,将创建闭包。 内部功能可以访问–

1. Its own variables.

1.自己的变量。

2. Outer function's variables and arguments.

2.外部函数的变量和参数。

Wait! Is this the definition of a closure or lexical scoping? Both definitions look the same. How they are different?

等待! 这是闭包或词汇作用域的定义吗? 两种定义看起来相同。 它们有何不同?

Well, that's why I defined lexical scoping above. Because closures are related to lexical/static scoping.

好吧,这就是为什么我在上面定义词法作用域的原因。 因为闭包与词汇/静态作用域有关。

Let’s again look at its other definition that will tell you how closures are different.

让我们再次查看其另一个定义,该定义将告诉您闭包有何不同。

Closure is when a function is able to access its lexical scope, even when that function is executing outside its lexical scope.
闭包是指某个函数能够访问其词法范围,即使该函数正在其词法范围之外执行。

Or,

要么,

Inner functions can access its parent scope, even after the parent function is already executed.
内部函数可以访问其父范围,即使已经执行了父函数也是如此。

Confused? Don't worry if you haven't yet gotten the point. I have examples to help you better understand. Let’s modify the first example of lexical scoping:

困惑? 如果您还没有明白这一点,请不要担心。 我有一些例子可以帮助您更好地理解。 让我们修改词法作用域的第一个示例:

function greetCustomer() {const customerName = "anchal";function greetingMsg() {console.log("Hi! " + customerName);}return greetingMsg;
}const callGreetCustomer = greetCustomer();
callGreetCustomer(); // output – Hi! anchal

The difference in this code is that we are returning the inner function and executing it later. In some programming languages, the local variable exists during the function’s execution. But once the function is executed, those local variables don’t exist and they will not be accessible.

这段代码的区别在于我们要返回内部函数并在以后执行它。 在某些编程语言中,局部变量在函数执行期间存在。 但是一旦函数执行完毕,这些局部变量将不存在,并且将无法访问。

Here, however, the scene is different. After the parent function is executed, the inner function (returned function) can still access the parent function's variables. Yes, you guessed right. Closures are the reason.

但是,这里的场景有所不同。 执行父函数后,内部函数(返回的函数)仍可以访问父函数的变量。 是的,你猜对了。 停业是原因。

The inner function preserves its lexical scope when the parent function is executing and hence, later that inner function can access those variables.

内部函数在执行父函数时会保留其词法范围,因此以后内部函数可以访问这些变量。

To get a better feel for it, let’s use the dir() method of the console to look into the list of the properties of callGreetCustomer:

为了更好地了解它,让我们使用控制台的dir()方法查看callGreetCustomer属性的callGreetCustomer

console.dir(callGreetCustomer);

From the above image, you can see how the inner function preserves its parent scope (customerName) when greetCustomer() is executed. And later on, it used customerName when callGreetCustomer() was executed.

从上面的图像中,您可以看到执行greetCustomer()时内部函数如何保留其父范围( customerName )。 之后,当执行callGreetCustomer()时,它使用了customerName

I hope this example helped you better understand the above definition of a closure. And maybe now you find closures a bit more fun.

我希望这个例子可以帮助您更好地理解闭包的上述定义。 也许现在您发现关闭更有趣了。

So what next? Let’s make this topic more interesting by looking at different examples.

那接下来呢? 通过查看不同的示例,使这个话题变得更加有趣。

实施中的关闭示例 (Examples of closures in action)

function counter() {let count = 0;return function() {return count++;};
}const countValue = counter();
countValue(); // 0
countValue(); // 1
countValue(); // 2

Every time you call countValue, the count variable value is incremented by 1. Wait – did you think that the value of count is 0?

每次调用countValue ,count变量的值都会增加1。等待–您是否认为count的值为0?

Well, that would be wrong as a closure doesn’t work with a value. It stores the reference of the variable. That’s why, when we update the value, it reflects in the second or third call and so on as the closure stores the reference.

嗯,这是错误的,因为闭包不能与值一起使用。 它存储变量的引用 。 这就是为什么当我们更新值时,它会在第二个或第三个调用中反映出来,依此类推,因为闭包存储了引用。

Feeling a bit clearer now? Let’s look at another example:

现在感觉更清晰了吗? 让我们看另一个例子:

function counter() {let count = 0;return function () {return count++;};
}const countValue1 = counter();
const countValue2 = counter();
countValue1();  // 0
countValue1();  // 1
countValue2();   // 0
countValue2();   // 1

I hope you guessed the right answer. If not, here is the reason. As countValue1 and countValue2, both preserve their own lexical scope. They have independent lexical environments. You can use dir() to check the [[scopes]] value in both the cases.

希望您猜对了答案。 如果没有,这就是原因。 与countValue1countValue2 ,它们都保留了自己的词汇范围。 他们有独立的词汇环境。 在这两种情况下,都可以使用dir()检查[[scopes]]值。

Let’s look at a third example.

让我们看第三个例子。

This one's a bit different. In it, we have to write a function to achieve the output:

这有点不同。 在其中,我们必须编写一个函数来实现输出:

const addNumberCall = addNumber(7);
addNumberCall(8) // 15
addNumberCall(6) // 13

Simple. Use your newly-gained closure knowledge:

简单。 使用您新获得的闭包知识:

function addNumber(number1) {return function (number2) {return number1 + number2;};
}

Now let’s look at some tricky examples:

现在让我们看一些棘手的例子:

function countTheNumber() {var arrToStore = [];for (var x = 0; x < 9; x++) {arrToStore[x] = function () {return x;};}return arrToStore;
}const callInnerFunctions = countTheNumber();
callInnerFunctions[0]() // 9
callInnerFunctions[1]() // 9

Every array element that stores a function will give you an output of 9. Did you guess right? I hope so, but still let me tell you the reason. This is because of the closure's behavior.

每个存储函数的数组元素都将为您提供9的输出。您猜对了吗? 希望如此,但还是让我告诉您原因。 这是由于闭包的行为。

The closure stores the reference, not the value. The first time the loop runs, the value of x is 0. Then the second time x is 1, and so on. Because the closure stores the reference, every time the loop runs it's changing the value of x. And at last, the value of x will be 9. So callInnerFunctions[0]() gives an output of 9.

闭包存储引用 ,而不是值。 循环第一次运行时,x的值为0。然后第二次循环x为1,依此类推。 因为闭包存储引用,所以每次循环运行时,它都会更改x的值。 最后,x的值为9。因此callInnerFunctions[0]()的输出为9。

But what if you want an output of 0 to 8? Simple! Use a closure.

但是,如果您想要0到8的输出怎么办? 简单! 使用闭包。

Think about it before looking at the solution below:

在查看以下解决方案之前,请考虑一下:

function callTheNumber() {function getAllNumbers(number) {return function() {return number;};}var arrToStore = [];for (var x = 0; x < 9; x++) {arrToStore[x] = getAllNumbers(x);}return arrToStore;
}const callInnerFunctions = callTheNumber();
console.log(callInnerFunctions[0]()); // 0
console.log(callInnerFunctions[1]()); // 1

Here, we have created separate scope for each iteration. You can use console.dir(arrToStore) to check the value of x in [[scopes]] for different array elements.

在这里,我们为每次迭代创建了单独的作用域。 您可以使用console.dir(arrToStore)检查[[scopes]]中x的值以获取不同的数组元素。

That’s it! I hope you can now say that you find closures interesting.

而已! 我希望您现在可以说您发现闭包很有趣。

To read my other articles, check out my profile here.

要阅读其他文章,请在此处查看我的个人资料。

翻译自: https://www.freecodecamp.org/news/javascript-closure-tutorial-with-js-closure-example-code/

javascript闭包

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/390263.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

1646. 获取生成数组中的最大值

1646. 获取生成数组中的最大值 给你一个整数 n 。按下述规则生成一个长度为 n 1 的数组 nums &#xff1a; nums[0] 0 nums[1] 1 当 2 < 2 * i < n 时&#xff0c;nums[2 * i] nums[i] 当 2 < 2 * i 1 < n 时&#xff0c;nums[2 * i 1] nums[i] nums[i …

docker保存日志文件到本地

其实很简单 docker logs 你需要添加的额外参数 容器id >文件名称 然后查看这个文件就可以了&#xff0c;也可以通过ftp协议下载到本地

防反射JavaScript –如何让您的JS等待

Debounce methods do not execute when invoked. Instead, they wait for a predetermined time before executing. If the same method is called again, the previous is cancelled and the timer restarts.防抖动方法在调用时不执行。 而是&#xff0c;它们在执行之前等待预…

21天学通Visual.Basic pdf

下载地址&#xff1a;网盘下载《21天学通Visual Basic(第2版)》是Visual Basic 6.0的入门教程&#xff0c;主要针对没有程序设计基础的读者&#xff0c;详细介绍了Visual Basic 6.0的基本概念和编程技术。书中通过大量的范例及综合练习来介绍Visual Basic 6.0的基本概念、语言特…

297. 二叉树的序列化与反序列化

297. 二叉树的序列化与反序列化 序列化是将一个数据结构或者对象转换为连续的比特位的操作&#xff0c;进而可以将转换后的数据存储在一个文件或者内存中&#xff0c;同时也可以通过网络传输到另一个计算机环境&#xff0c;采取相反方式重构得到原数据。 请设计一个算法来实现…

交互式图表_如何构建罗马数字转换器和交互式罗马数字图表

交互式图表The Roman numerals are no longer an essential part of our daily lives. But we do use them when designing monuments, clocks, and even for sporting events.罗马数字不再是我们日常生活中必不可少的部分。 但是我们在设计纪念碑&#xff0c;钟表甚至体育赛事…

Python 08 面向对象

Python 面向对象 1、编程范式 2、面向对象特性 3、属性、方法 4、三大特性 5、高级方法 6、类的特殊成员方法 7、反射 8、异常处理 9、单例模式 一、编程范式 编程&#xff1a;程序员用特定的语法数据结构算法组成的代码来告诉计算机如何执行任务的过程 &#xff0c; 实现一个…

eclipse手动添加SVN插件

https://www.cnblogs.com/hcl1991/p/5888461.html 1.手动下载svn插件&#xff08;百度SVNsite-1.8.18&#xff09; 2.将下载好的SVNsite-1.8.18.zip 解压 3.在eclipse安装目录的plugins新建SVN文件夹 4.将SVNsite-1.8.18解压包下的features和plugins拷贝到新建的SVN文件夹下 5.…

440. 字典序的第K小数字

440. 字典序的第K小数字 给定整数 n 和 k&#xff0c;找到 1 到 n 中字典序第 k 小的数字。 注意&#xff1a;1 ≤ k ≤ n ≤ 109。 示例 : 输入: n: 13 k: 2 输出: 10 解释: 字典序的排列是 [1, 10, 11, 12, 13, 2, 3, 4, 5, 6, 7, 8, 9]&#xff0c;所以第二小的数字是…

微信小程序 设置背景占满整个页面

微信小程序,默认的根节点是<page></page>&#xff0c;所有内容都包裹在这个标签里&#xff0c;如何让页面的背景占满整个屏幕&#xff1f;&#xff1f; <page><view class"bg">....</view> </page> .bg {background-image:ur…

udemy下载课程无法播放_最好的Udemy Web开发课程+热门免费课程

udemy下载课程无法播放Heres a list of some of the most popular web development courses on Udemy:以下是Udemy上一些最受欢迎的Web开发课程的列表&#xff1a; Best General Web Development Courses on Udemy:关于Udemy的最佳常规Web开发课程&#xff1a; The Complete …

渗透测试初学者_渗透测试许可证:面向初学者的道德黑客课程

渗透测试初学者A penetration test is an authorized cyberattack on a computer system, performed to evaluate the security of the system. 渗透测试是对计算机系统的授权网络攻击&#xff0c;旨在评估系统的安全性。 Weve released a full pentesting course on the free…

Codeforces 915 E Physical Education Lessons

题目描述 This year Alex has finished school, and now he is a first-year student of Berland State University. For him it was a total surprise that even though he studies programming, he still has to attend physical education lessons. The end of the term is …

JDK 11 还有一个处于计划阶段的 JEP:让其支持 TLS 1.3

开发四年只会写业务代码&#xff0c;分布式高并发都不会还做程序员&#xff1f; >>> JDK 11 最近有什么消息&#xff1f;我们不妨来看一下它的进展情况&#xff0c;包括最新的 JEP 提案。Java 的新版本发布计划意味着总会有一款新的 JDK 即将推出。根据他们的计划&a…

498. 对角线遍历

498. 对角线遍历 给定一个含有 M x N 个元素的矩阵&#xff08;M 行&#xff0c;N 列&#xff09;&#xff0c;请以对角线遍历的顺序返回这个矩阵中的所有元素&#xff0c;对角线遍历如下图所示。 示例: 输入: [ [ 1, 2, 3 ], [ 4, 5, 6 ], [ 7, 8, 9 ] ] 输出: [1,2,4,7,5…

软件测试测试用例编写_不要先编写所有软件测试-只需编写一个

软件测试测试用例编写Test Driven Development (TDD) is sometimes described as “writing tests first”. The TDD mantra states that we should not write code before we have written automated tests that exercise that code. Writing code first is considered subopt…

练习五

1.团队模式和团队的开发模式有什么关系&#xff1f; 答&#xff1a; 首先我来解释一下这两个名词&#xff1a; 我查资料了解了一下&#xff0c;团队模式&#xff0c;更偏向于多人合作的那种&#xff0c;而且我理解的“团队”会是一种多人合作的情况下&#xff0c;长期磨合后的一…

Squid 访问控制配置

Squid 访问控制配置 主配置文件内加入限制参数 vim /etc/squid/squid.conf # 访问控制 acl http proto HTTP # 限制访问 good_domain添加两个域名“.”表示半解析(相当于*) acl good_domain dstdomain .kevin.net .baidu.com # 允许good_domain内的域名访问 http_access allow …

剑指 Offer 62. 圆圈中最后剩下的数字

0,1,,n-1这n个数字排成一个圆圈&#xff0c;从数字0开始&#xff0c;每次从这个圆圈里删除第m个数字&#xff08;删除后从下一个数字开始计数&#xff09;。求出这个圆圈里剩下的最后一个数字。 例如&#xff0c;0、1、2、3、4这5个数字组成一个圆圈&#xff0c;从数字0开始每…

rust 编程入门_面向初学者的Rust –最受欢迎的编程语言入门

rust 编程入门Rust has been voted Stack Overflow’s most loved programming language for five years in a row. This article will tell you why Rust is awesome.Rust已连续五年被评为Stack Overflow最受欢迎的编程语言。 本文将告诉您为什么Rust很棒。 Rust is a system…