JavaScript代码执行的先后顺序问题


Posted in Javascript onOctober 29, 2017

一、js--->单线程

 

严格意义上来说,javascript没有多线程的概念,所有的程序都是单线程依次执行的。

  1、什么是单线程?

通俗点说,就是代码在执行过程中,另一段代码想要执行就必须等当前代码执行完成后才可以进行。我们拿一段代码来解释一下吧

for(var i=1;i<=3;i++){
   setTimeout(function(){
     console.log(i); //输出:4,4,4
   },0)
 }

我们来看一下上面的这段代码,既然延时器时间设置为0,那么应该执行一遍循环就应该立即打印出一个i,但是最终的打印结果为:4,4,4。之所以会出现上面的结果,正是因为js代码是单线程应用。

在执行过程中,先遇到for循环,for循环先进入线程。当i=1时,循环走到setTimeOut后,此时的for循环还没有执行完成,setTimeOut就会被放入一个地方(线程池)等待执行。此时for循环继续执行,当i=2时,for循环仍没有执行完,这时的setTimeOut仍会被放在线程池中等待执行……依次类推,直到for循环转完3遍后,for循环执行完了,此时线程空闲了,在线程池中等待执行的setTimeOut依次执行打印i,而for循环执行完成后,i变成了4,所以打印出了三个4。

2、如果想要改变上面的状况可以运用以下代码

//将var变为let
for(let i=1; i<=3; i++){
  setTimeOut(function(){
    console.log(i); //输出的结果为1,2,3
  },0);
}
//用自执行函数进行包裹
for(var i=1; i<=3; i++){
  !function(i){
    setTimeOut(function(){
      console.log(i); //输出的结果为1,2,3
    },0);
  }(i)
}

二、js中的函数作用域和代码的执行

>>>函数作用域

我们先来了解一下以下几个概念:

1、在js语言中,没有类似于c语言这样的块级作用域。

2、js语言中的顶级作用域为window对象范围内,称为全局作用域,在全局作用域中声明的变量为全局变量。

3、js函数范围内的变量只能在函数内部使用,函数外部无法使用,这样的变量为局部变量。

4、js函数可以嵌套,多个函数的嵌套构成了作用域的层层嵌套,这称为js中的作用域链。

5、js作用域链变量访问规则:

(1)、当前作用域内存在要访问的变量时,则使用当前作用域中的变量。

 

    (2)、当前作用域中不存在要访问的变量时,则会到上一层作用域中寻找,直到全局作用域。

>>>执行顺序

 

js代码执行分为两个部分:

1、代码的检查装载阶段(预编译阶段),此阶段进行变量和函数的声明,但是不对变量进行赋值,变量的默认值为undefined。

2、代码的执行阶段,此阶段对变量进行赋值和函数的声明。

看了上面的一些具体的概念,我们拿一段代码进行举例说明:

var a=1; //声明了一个全局变量
function func(){
  console.log(a); //输出:undefined。打印a,而在func这个作用域中已经声明了a变量,按照js的执行顺序,此时的a并未被赋值。
  var a=1;
  console.log(a); //输出:1。
}
func();

看上面的代码:第一个a输出undefined。原因:js作用域链的访问规则,当前作用域内存在要访问的变量a,所以使用当前作用域中的变量。再根据js代码的执行顺序,此时的a只是声明了而并未被赋值,默认为undefined,所以输出undefined。

而第二个a,输出1,正是因为此时的a已经被声明且被赋值,所以a输出1。

总结

以上所述是小编给大家介绍的JavaScript代码执行的先后顺序问题,希望对大家有所帮助,如果大家有任何疑问欢迎给我留言,小编会及时回复大家的!

Javascript 相关文章推荐
JavaScript自定义DateDiff函数(兼容所有浏览器)
Mar 01 Javascript
js 与 php 通过json数据进行通讯示例
Mar 26 Javascript
js在IE与firefox的差异集锦
Nov 11 Javascript
jQuery元素的隐藏与显示实例
Jan 20 Javascript
jQuery插件制作之全局函数用法实例
Jun 01 Javascript
用NODE.JS中的流编写工具是要注意的事项
Mar 01 Javascript
vue使用better-scroll实现下拉刷新、上拉加载
Nov 23 Javascript
jQuery事件多次绑定与解绑问题实例分析
Feb 19 jQuery
js+canvas实现纸牌游戏
Mar 16 Javascript
JS co 函数库的含义和用法实例总结
Apr 08 Javascript
jQuery实现朋友圈查看图片
Sep 11 jQuery
vue 函数调用加括号与不加括号的区别
Oct 29 Javascript
springmvc接收jquery提交的数组数据代码分享
Oct 28 #jQuery
ES6中Class类的静态方法实例小结
Oct 28 #Javascript
jQuery实现的简单前端搜索功能示例
Oct 28 #jQuery
jQuery实现点击下拉框中的值累加到文本框中的方法示例
Oct 28 #jQuery
jquery ajax异步提交表单数据的方法
Oct 27 #jQuery
mint-ui的search组件在键盘显示搜索按钮的实现方法
Oct 27 #Javascript
基于BootStrap的文本编辑器组件Summernote
Oct 27 #Javascript
You might like
php用数组返回无限分类的列表数据的代码
2010/08/08 PHP
PHP第一季视频教程(李炎恢+php100 不断更新)
2011/05/29 PHP
thinkphp浏览历史功能实现方法
2014/10/29 PHP
分享最受欢迎的5款PHP框架
2014/11/27 PHP
Yii2中如何使用modal弹窗(基本使用)
2016/05/30 PHP
关于火狐(firefox)及ie下event获取的两种方法
2012/12/27 Javascript
jQuery仿Excel表格编辑功能的实现代码
2013/05/01 Javascript
js jquery ajax的几种用法总结(及优缺点介绍)
2014/01/28 Javascript
封装了一个支持匿名函数的Javascript事件监听器
2014/06/05 Javascript
javascript实现根据身份证号读取相关信息
2014/12/17 Javascript
实例讲解JS中setTimeout()的用法
2016/01/28 Javascript
jQ处理xml文件和xml字符串的方法(详解)
2016/11/22 Javascript
JS 实现可停顿的垂直滚动实例代码
2016/11/23 Javascript
Bootstrap基本组件学习笔记之进度条(15)
2016/12/08 Javascript
Nodejs基于LRU算法实现的缓存处理操作示例
2017/03/17 NodeJs
vue v-on监听事件详解
2017/05/17 Javascript
input 标签实现输入框带提示文字效果(两种方法)
2017/10/09 Javascript
在一个页面实现两个zTree联动的方法
2017/12/20 Javascript
详解在vue-test-utils中mock全局对象
2018/11/07 Javascript
elementUI Tree 树形控件的官方使用文档
2019/04/25 Javascript
javascript原型链学习记录之继承实现方式分析
2019/05/01 Javascript
js实现圆形显示鼠标单击位置
2020/02/11 Javascript
Vue自定义多选组件使用详解
2020/09/08 Javascript
[03:14]DOTA2斧王 英雄基础教程
2013/11/26 DOTA
python模拟新浪微博登陆功能(新浪微博爬虫)
2013/12/24 Python
Python实现的飞速中文网小说下载脚本
2015/04/23 Python
Python中文竖排显示的方法
2015/07/28 Python
flask + pymysql操作Mysql数据库的实例
2017/11/13 Python
SKECHERS斯凯奇中国官网:来自美国的运动休闲品牌
2018/11/14 全球购物
酒店管理自荐信
2013/10/23 职场文书
三年大学生活自我鉴定
2014/01/21 职场文书
外国人聘用意向书
2014/04/01 职场文书
供货协议书
2014/04/22 职场文书
2014年银行员工年终自我评价
2014/09/19 职场文书
红色影片观后感
2015/06/18 职场文书
详解Nginx启动失败的几种错误处理
2021/04/01 Servers