1. JS编译解析的流程

1.1 JS运行分三步

语法分析(通篇扫描是否有语法错误),预编译(发生在函数执行的前一刻),解释执行(一行行执行)。

1.2 预编译执行分五步

  1. 创建AO对象(Activation Object  执行期上下文)

  2. 找形参和变量声明,将变量和形参名作为AO属性名,值为undefined. 变量声明提升(变量放到后面也不会报错,只是未定义类型)如:console.log(a);var a=10;结果undenfined;

  3. 将实参值和形参统一(传参)

  4. 在函数体里面找到函数声明{函数声明整体提升(相当于放到程序最前面)}

  5. 值赋予函数体,执行(声明函数和变量的部分直接不看了)

2. 函数作用域和作用域链

2.1 函数作用域

每个javascript函数都是一个对象,对象中有的属性可以访问,有的不能,这些属性仅供javascript引擎存取,如[[scope]]。

[[scope]]就是函数的作用域,其中存储了执行期上下文的集合。

执行期上下文: 当函数执行时,会创建一个称为执行期上下文的内部对象(AO)。一个执行期上下文定义了一个函数执行时的环境,函数每次执行时对应的执行期上下文都是独一无二的,所以多次调用一个函数会导致创建多个执行期上下文,当函数执行完毕,它所产生的执行上下文被销毁。

2.2 作用域链

[[scope]]中所存储的执行期上下文对象的集合,这个集合呈链式链接,我们称这种链式链接为作用域链。查找变量时,要从作用域链的顶部开始查找。Activation Object(AO)到Global Object(GO)。

2.3 闭包

[!NOTE]

当内部函数被保存到外部时,将会生成闭包。生成闭包后,内部函数依旧可以访问其所在的外部函数的变量。

闭包问题的解决方法:立即执行函数、let

2.3.1 详细解释

  1. 当函数执行时,会创建一个称为执行期上下文的内部对象(AO),执行期上下文定义了一个函数执行时的环境。

  2. 函数还会获得它所在作用域的作用域链,是存储函数能够访问的所有执行期上下文对象的集合,即这个函数中能够访问到的东西都是沿着作用域链向上查找直到全局作用域。

  3. 函数每次执行时对应的执行期上下文都是独一无二的,当函数执行完毕,函数都会失去对这个作用域链的引用,JS的垃圾回收机制是采用引用计数策略,如果一块内存不再被引用了那么这块内存就会被释放。

  4. 但是,当闭包存在时,即内部函数保留了对外部变量的引用时,这个作用域链就不会被销毁,此时内部函数依旧可以访问其所在的外部函数的变量,这就是闭包。

2.3.2 闭包的应用

先看两个例子,两个例子都打印5个5

for (var i = 0; i < 5; i++) {
setTimeout(function timer() {
console.log(i)
}, i * 100)
}
function test() {
var a = [];
for (var i = 0; i < 5; i++) {
a[i] = function () {
console.log(i);
}
}
return a;
} var myArr = test();
for(var j=0;j<5;j++)
{
myArr[j]();
}

解决方法: 使用立即执行函数

for (var i = 0; i < 5; i++) {
;(function(i) {
setTimeout(function timer() {
console.log(i)
}, i * 100)
})(i)
}
function test(){
var arr=[];
for(i=0;i<10;i++)
{
(function(j){
arr[j]=function(){
console.log(j);
}
})(i)
}
return arr;
} var myArr=test();
for(j=0;j<10;j++)
{
myArr[j]();
}

2.3.3 闭包-封装私有变量

function Counter() {
let count = 0;
this.plus = function () {
return ++count;
}
this.minus = function () {
return --count;
}
this.getCount = function () {
return count;
}
} const counter = new Counter();
counter.puls();
counter.puls();
console.log(counter.getCount())

3. 作用域与变量声明提升?

  1. 在 JavaScript 中,函数声明与变量声明会被 JavaScript 引擎隐式地提升到当前作用域的顶部
  2. 声明语句中的赋值部分并不会被提升,只有名称被提升
  3. 函数声明的优先级高于变量,如果变量名跟函数名相同且未赋值,则函数声明会覆盖变量声明
  4. 如果函数有多个同名参数,那么最后一个参数(即使没有定义)会覆盖前面的同名参数

4. 构造函数,new时发生了什么?

   var obj  = {};
obj.__proto__ = Base.prototype;
Base.call(obj);
  1. 创建一个新的对象 obj;
  2. 将这个空对象的__proto__成员指向了Base函数对象prototype成员对象
  3. Base函数对象的this指针替换成obj, 相当于执行了Base.call(obj);
  4. 如果构造函数显示的返回一个对象,那么则这个实例为这个返回的对象。 否则返回这个新创建的对象

5. 函数参数是对象会发生什么问题?

function test(person) {
person.age = 26
person = {
name: 'yyy',
age: 30
} return person
}
const p1 = {
name: 'hy',
age: 25
}
const p2 = test(p1)
console.log(p1) // -> {name: "hy", age: 20}
console.log(p2) // -> {name: "yyy", age: 30}

person = {} 这一步操作就将应用与原来的分离了

6. JavaScript 中,调用函数有哪几种方式?

  1. 方法调用模式 Foo.foo(arg1, arg2);
  2. 函数调用模式 foo(arg1, arg2);
  3. 构造器调用模式 (new Foo())(arg1, arg2);
  4. call/applay 调用模式 Foo.foo.call(that, arg1, arg2);
  5. bind 调用模式 Foo.foo.bind(that)(arg1, arg2)();

最新文章

  1. Angular2学习笔记——NgModule
  2. IBatisNet使用教程
  3. Linux Shell变量
  4. jQuery实现在线文档
  5. 联想扬天 电脑 键盘改默认fn功能键
  6. webAPI 自动生成帮助文档
  7. 中小企业 IT 运维福利:快速构建 on-call 机制
  8. 搭建hadoop环境,在win7的eclipse上远程操作(Linux上)hadoop2.6.0出错的一些总结
  9. 图论(网络流):[HNOI 2013]切糕
  10. Winform中上传、下载文件选择打开文件的位置
  11. 为什么jQuery要返回jQuery.fn.init对象
  12. url_for()中的坑,url_for操作对象是函数,而不是route里的路径
  13. Android 音视频编解码——RGB与YUV格式转换
  14. [BZOJ]1005 明明的烦恼(HNOI2008)
  15. Promise(避免金字塔回调)
  16. [UE4]记录瞬移目标点
  17. ISCC的 Web——WP
  18. Nginx——debug的使用
  19. vue 钩子
  20. 【bzoj1855】 [Scoi2010]股票交易 单调队列优化DP

热门文章

  1. 如何在Mac上使用Netstat命令
  2. 【bzoj4555】[Tjoi2016&amp;Heoi2016]求和(NTT+第二类斯特林数)
  3. sass参考手册
  4. [CodeForces-1225A] Forgetting Things 【构造】
  5. IT兄弟连 HTML5教程 CSS3揭秘 CSS选择器2
  6. python爬虫-京东商品爬取
  7. JMeter命令行执行+生成HTML报告
  8. SSM框架之Spring(5)JdbcTemplate及spring事务控制
  9. 【Gradle】Android Gradle 多项目构建
  10. DevOps VS 职责分离