JavaScript解析机制与闭包原理实例详解
js代码解析机制:
js代码解析之前会创建一个如下的词法环境对象(仓库):LexicalEnvironment{ }
在扫描js代码时会把:
1、用声明的方式创建的函数的名字;
2、用var定义的变量的名字存到这个词法环境中;
3、同名的时候:函数声明会覆盖变量,下面的函数声明会覆盖上面的同名函数;
4、函数的值为:对函数的一个引用; 变量的值为undefined;
5、如果用函数表达式的方式创建一个函数:
var fn = function(){ } 这样词法环境中存的是一个变量名fn,并赋值为undefined;
在调用函数的时候如果在函数上面调用就会出现和变量一样的情况报错undefined;
这也是以两种不同方式创建函数的区别;
| 
 1 
2 
3 
4 
5 
 | 
LexicalEnvironment(这个词法环境===window){fn: 对函数的一个引用;b:undefined;} | 
用声明的方式创建的函数:
| 
 1 
 | 
function fn(){ }; | 
用var定义的变量:
| 
 1 
 | 
var b=5; | 
每次调用函数的时候就会创建一个新的词法环境对象(仓库):LexicalEnvironment{ };
在解析函数内部的变量和函数声明的时候跟全局词法环境相同,不过有两点需要注意,如下:
| 
 1 
2 
3 
4 
5 
 | 
LexicalEnvironment(这个词法环境===fn){a:对函数的一个引用;(解析的时候函数声明把变量覆盖了,尽管变量已经被赋值为1)b:2;(解析的时候把变量存在了词法环境里,同时赋值为2)} | 
| 
 1 
2 
3 
4 
5 
6 
7 
 | 
function fn (a,b){  alert(a)// function a(){ }  alert(b)//2  var b= 100;  function a(){ }}fn(1,2); | 
调用函数并传递参数的时候,词法环境里会再存变量名的同时赋值,如果是函数内部有同名的函数声明则会把传入的参数覆盖;如果形参只有一个,那么另一个实参则被赋值为undefined;
闭包:
定义:(有多种定义)
1、(比较通俗的定义):函数嵌套函数,内部函数可以引用外部函数的参数和变量,这些参数和变量不会被垃圾回收机制所回收;
2、在计算机科学中,闭包是词法闭包的简称,是引用了自由变量的函数,这个被引用的自由变量将和这个函数一同存在,即使已经离开了创造它的环境也不例外(意思就是不会被销毁)。
3、闭包是由函数和其相关的引用环境组合而成的实体。(潜台词就是这个函数将和引用环境同时存在,必须有引用)
综合来说,不管怎么定义都是在围绕着两个本质:函数在引用变量,这个变量将不会被销毁。
什么叫做被引用的自由变量离开了创造它的环境?如下:
| 
 1 
2 
3 
4 
5 
6 
7 
8 
9 
 | 
function fn(){  var a = 10;  var b = 20;  return function fn2(){    alert(a);  }}var result = fn();result();//10; | 
以上代码就是fn2在被return出去以后,离开了fn函数这个环境,但是在外部调用依然能够访问到fn的变量;
这就是被引用的变量不会被销毁;同理在自执行函数这个闭包里,虽然自执行函数在自身执行过后内部变量本该被垃圾回收机制所回收,但是由于其内部有引用它的变量的子函数,也就是说构成了闭包,它的变量依然不会被销毁;
由此可见闭包的一个作用就是:我们能够通过闭包的方法来在外部访问到一个内部函数的变量;
很多人在解释闭包的时候都会把子函数return出去以后在外部调用,其实无论在哪里调用,闭包都已经形成了,只要是函数嵌套函数,并且子函数引用了父函数的变量,(不论子函数有没有被调用,这个用一种方法证明:在子函数内部打断点,在f12中观察闭包里的内容,已经出现了引用函数,这时候调用还没有被执行)这个时候闭包已经形成了。
jQuery插件闭包的本质:就是形成了作用域链。
注意:形成闭包的条件:1、函数要嵌套;2、子函数要引用父函数的变量(如果没有引用,则不会形成闭包,如果是引用父函数的父函数的变量也会形成闭包);
父函数每调用一次,就会形成一个新的闭包(函数每调用一次,就 会复制一份),也就是说形成一个新的词法作用域,重新引用父函数的变量;
以下代码说明:
| 
 1 
2 
3 
4 
5 
6 
7 
8 
9 
10 
11 
12 
13 
 | 
function fn() {  var num = 1;  return function() {    num++;    alert(num);  }}var result1 = fn();result1();//2;result1();//3;var result2 = fn();result2();//2;result2();//3; | 
在fn被调用两次时,都形成了新的闭包,有各自新的词法作用域,所以result2的输出结果不受result1的影响;
函数每调用一次,都会复制一份新的,可以说明for循环里i的问题;
| 
 1 
2 
3 
4 
5 
6 
7 
 | 
for (var i = 0; i < 5; i++) {  (function(index){    oDiv.onclick = function() {      alert(index)    }  })(i)} | 
自执行函数每调用一次都会复制一份新的,传进的i值也在变化,由于在函数静态作用域里,在预解析阶段已经确定了变量的作用域,所以子函数引用的父函数变量index只能是每次复制的那个父函数变量,所以就实现了我们想要取不同的i值的目的;
图片延迟加载闭包的用途:
1、匿名自执行函数
不污染全局变量,(否则被声明为全局变量的话别的函数可能误用这些变量;造成全局对象过于庞大,影响访问速度(因为变量的取值是需要从原型链上遍历的)。提高效率;
2、结果缓存:
我们开发中会碰到很多情况,设想我们有一个处理过程很耗时的函数对象,每次调用都会花费很长时间利用闭包,它不会释放外部的引用,从而函数内部的值可以得以保留。,这样我们在第二次调用的时候,就会从缓存中读取到该资源。
3、实现封装;
4、实现类和继承(构造函数);
JavaScript解析机制与闭包原理实例详解的更多相关文章
- javascript解析机制、闭包详解
		
js解析机制: js代码解析之前会创建一个如下的词法环境对象(仓库):LexicalEnvironment{ } 在扫描js代码时会把: 1.用声明的方式创建的函数的名字: 2.用var定义的变量的名 ...
 - python描述符(descriptor)、属性(property)、函数(类)装饰器(decorator )原理实例详解
		
1.前言 Python的描述符是接触到Python核心编程中一个比较难以理解的内容,自己在学习的过程中也遇到过很多的疑惑,通过google和阅读源码,现将自己的理解和心得记录下来,也为正在为了该问题 ...
 - php单点登录实现原理实例详解
		
单点登录SSO(Single Sign On)说得简单点就是在一个多系统共存的环境下,用户在一处登录后,就不用在其他系统中登录,也就是用户的一次登录能得到其他所有系统的信任. 单点登录在大型网站里使用 ...
 - 转载 《AngularJS》5个实例详解Directive(指令)机制
		
<AngularJS>5个实例详解Directive(指令)机制 大漠穷秋 本文整理并扩展了<AngularJS>这本书第六章里面的内容,此书近期即将由电子工业出版社出版,敬请 ...
 - 我的书籍《深入解析Java编译器:源码剖析与实例详解》就要出版了
		
一个十足的技术迷,2013年毕业,做过ERP.游戏.计算广告,在大公司呆过,但终究不满足仅对技术的应用,在2018年末离开了公司,全职写了一本书<深入解析Java编译器:源码剖析与实例详解> ...
 - javascript常用经典算法实例详解
		
javascript常用经典算法实例详解 这篇文章主要介绍了javascript常用算法,结合实例形式较为详细的分析总结了JavaScript中常见的各种排序算法以及堆.栈.链表等数据结构的相关实现与 ...
 - JavaScript学习笔记-实例详解-类(一)
		
实例详解-类(一): //每个javascript函数(除了bind())都自动拥有一个prototype对象// 在未添加属性或重写prototype对象之前,它只包含唯一一个不可枚举属性const ...
 - 041——VUE中组件之pros数据的多种验证机制实例详解
		
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
 - javascript解析机制——预解析
		
JavaScript解析机制是什么? JavaScript解析过程分为两个阶段,一个是编译阶段,另外一个就是执行阶段. * 编译阶段 编译阶段就是我们常说的JavaScript预解析( ...
 
随机推荐
- Akka-CQRS(6)- read-side
			
前面我们全面介绍了在akka-cluster环境下实现的CQRS写端write-side.简单来说就是把发生事件描述作为对象严格按发生时间顺序写入数据库.这些事件对象一般是按照二进制binary方式如 ...
 - 一文读懂高性能网络编程中的I/O模型
			
1.前言 随着互联网的发展,面对海量用户高并发业务,传统的阻塞式的服务端架构模式已经无能为力.本文(和下篇<高性能网络编程(六):一文读懂高性能网络编程中的线程模型>)旨在为大家提供有用的 ...
 - 虚拟机安装mysql
			
安装mysql8.0方法: 1.首先,ubuntu安装mysql8.0(此版本比较好装),所以要先更新一下源,去MySQL官网http://dev.mysql.com/downloads/repo/ ...
 - Kali学习笔记13:操作系统识别
			
为什么要扫描操作系统呢? 其实和上一篇博客:<服务扫描>类似,都是为了能够发现漏洞 发现什么漏洞? 不同的操作系统.相同操作系统不同版本,都存在着一些可以利用的漏洞 而且,不同的系统会默认 ...
 - kafka扫盲笔记,实战入门
			
Kafka作为大数据时代的产物,自有其生存之道.让我们跟随扫盲班的培训,进行大致了解与使用kafka吧.(平时工作有使用不代表就知道kafka了哟) 1. kafka介绍 1.1. 拥有的能力(能干什 ...
 - python高级-面向对象(11)
			
一.面向过程和面向对象 面向过程:根据业务逻辑从上到下写代码 面向对象:将数据与函数绑定到一起,进行封装,这样能够更快速的开发程序,减少了重复代码的重写过程 二.类和对象 1.类的概念 面向对象编程的 ...
 - 导入项目报错【Minimum supported Gradle version is 3.3. Current version is 2.14.1】
			
问题描述 导入项目的时候,因为同事的开发环境是Android Studio 2.3.2 Gradle3.3.而我的开发环境是Android Studio 2.2.2 Gradle2.14.1. 所以 ...
 - 【原创】驱动枚举之QueryServiceStatus
			
BOOL WINAPI QueryServiceStatus( _In_ SC_HANDLE hService, _Out_ LPSERVICE_STATUS lpServiceStatus ); 函 ...
 - Python数据科学“冷门”库
			
Python是一种神奇的语言.事实上,它是近几年世界上发展最快的编程语言之一,它一次又一次证明了它在开发工作和数据科学立场各行业的实用性.整个Python系统和库是对于世界各地的用户(无论是初学者或者 ...
 - java中Char到底是什么格式的编码
			
文本处理中经常有这样的逻辑: String s = new String(bts, "UTF-8"); 看String源代码,里面是一个char[],将bts按照某种编码方式,变成 ...