浅谈JavaScript变量声明提升
前段时间阿里实习生内推,一面就被刷了,也是郁闷。今天系统给发通知,大致意思就是内推环节不足以了解彼此,还可以参加笔试,于是赶紧再投一次。官网流程显示笔试时间3月31日,时间快到了,开始刷题。网上搜了一下去年题目,我擦,第一题就不会(伤心中...),继续往下看,第四题也不会。。。好嘛,搜答案吧。搜了一个多小时的博客,结合自己的理解,写下了这篇博文。。。
2014年阿里巴巴前端工程师笔试题目1:
var v = "Hello world";
(function() {
console.log(v);
var v = "I love you";
bar();
foo();
function bar(){
console.log("this is bar!");
} var foo = function() {
console.log("this is foo!");
}; })();
请写出上面的运行结果并说出背后的理由。
题目4:指出一下代码的运行结果,并解释为什么。
function bar() {
return foo;
foo = 10;
function foo() {}
var foo = 11;
}
alert(typeof bar());
其实上面两道题考察的核心是一样的,即JavaScript变量声明的相关概念,之前了解过一点,但了解的不是太深。
要谈JavaScript变量声明,就离开不了作用域的问题。而JavaScript中没有类似C,JAVA等语言中的块级作用域。如:
#include <studio.h>
int main() {
int i = 1;
printf("d%",i); //1
if(1){
int i = 2;
printf("d%",i); //2
}
printf("d%",i); //1
return 0;
}
而JavaScript中相应的输出结果就变成了1,2,2。原因就是JavaScript中只有函数作用域,这也是导致JavaScript中一个经典问题的原因,即循环遍历DOM节点绑定事件后触发问题。如:
window.onload = function() {
var oLi = document.getElementsByTagName("li");
for(var i=0,len=oLi.length;i<len;i++){
oLi[i].onclick = function() {
alert(i);
};
}
};
扯得有点远,下面还是说一下JavaScript语言中的变量声明提升的问题吧。
1、当访问函数内的变量时,JavaScript会按照下面顺序查找:
(1)语言级别:默认在所用作用域下的this,arguments
(2)传入参数:函数命名的参数,作用域是当前函数体。
(3)函数声明:如题1中的function bar() {} 与 题4中的function foo() {}
(4)变量声明:如var v 与 var foo
2、变量声明与函数声明经常被JavaScript引擎隐式的提升到当前作用域的顶部。
2.1、变量声明:
局部作用域内声明变量需要加关键词var,即
function a(){
console.log(typeof b);
var b = 3;
}
实际上被解释成
function a(){
var b;
console.log(typeof b);
b = 3;
}
而如果不加关键词var,则默认引用全局作用域内的相关变量,浏览器中即window作用域。如图:
所以如果你能理解下面代码的运行结果是a=10,则说明对变量声明部分已经足够了解。
<span style="white-space:pre"> </span>var a = 1;
function b() {
if(!a) {
var a = 10;
}
alert(a);
}
b();
2.2、函数声明:我们都知道函数命名有两种方式,一种是函数声明方式function bar() {} 另外一种就是通过赋值语句的方式,即函数声明表达式var foo = function() {};
而这两种函数命名方式虽然都会出现变量声明提升的现象,但第一种提升现象会包含函数体部分一起被提升,即整个函数bar被提升。而第二种则只是简单的变量提升(想象一下将赋值语句后面的函数代码改为基本变量,如数字5,则和上一步所说的变量声明部分完全吻合)。
因此得出如下结论:
(1)JavaScript中声明语句会被提升,而赋值语句不会被提升。
(2)函数声明提升优先于变量声明提升。
现在你应该能明白阿里笔试题1中输出结果是undefined this is bar! error:undefined is not a function. 题4中的结果是function.
特此感谢以下博主的精彩博文:
浪迹天涯《JavaScript中变量提升------Hoisting》
Nomospace《翻译——JavaScript中的变量作用域与变量声明提升》
浅谈JavaScript变量声明提升的更多相关文章
- JavaScript 变量声明提升
JavaScript 变量声明提升 一.变量提升的部分只是变量的声明,赋值语句和可执行的代码逻辑还保持在原地不动 二.在基本的语句(或者说代码块)中(比如:if语句.for语句.while语句.swi ...
- javascript变量声明提升和函数声明提升
在ES6之前,JavaScript没有块级作用域(一对花括号{}即为一个块级作用域),只有全局作用域和函数作用域.变量提升即将变量声明提升到它所在作用域的最开始的部分. JS的解析过程分为两个阶段:预 ...
- JavaScript变量声明提升
JavaScript代码在被解析引擎执行前,会被“编译”把变量声明等放在合适的作用域中,如果不了解这一点,会让人产生很多疑惑. 文章:详解js变量声明提升
- 浅谈JS变量声明和函数声明提升
先来两个问题 很多时候,在直觉上,我们都会认为JS代码在执行时都是自上而下一行一行执行的,但是实际上,有一种情况会导致这个假设是错误的. a = 2; var a; console.log(a); 按 ...
- 谈javascript变量声明
之前的面试中遇到过一道面试题 var a =10;(function(){ console.log(a); var a =20;})() 短短5行代码log的结果是什么? 如果把var a = 20; ...
- javascript变量声明提升(hoisting)
javascript的变量声明具有hoisting机制,JavaScript引擎在执行的时候,会把所有变量的声明都提升到当前作用域的最前面. 先看一段代码 1 2 3 4 5 var v = &quo ...
- javascript变量声明及作用域总结
javascript变量声明及作用域总结 一.总结 一句话总结:还是得好好看书,光看视频是不得行的,浅学无用,要相互印证,要真正理解才有用,比如<Javascript权威指南> 书 1.j ...
- javascript变量声明 及作用域
javascript变量声明提升(hoisting) http://openwares.net/js/javascript_declaration_hoisting.html 可能要FQ一下 java ...
- 【转】javascript变量声明 及作用域
javascript变量声明提升(hoisting) javascript的变量声明具有hoisting机制,JavaScript引擎在执行的时候,会把所有变量的声明都提升到当前作用域的最前面. 先看 ...
随机推荐
- 从jscript脚本混淆说起
转载:http://www.freebuf.com/column/144897.html 脚本病毒是一个一直以来就存在,且长期活跃着的一种与PE病毒完全不同的一类病毒类型,其制作的门槛低.混淆加密方式 ...
- React中input框设置value报错解析
react input 不设置onChange的常见错误截图 表单是前端非常重要的一块内容,并且往往包含了错误校验等逻辑. React对表单元素做了专门的优化处理,他对表单元素做了一些抽象,使得他们 ...
- 使用log4j2分离系统日志与业务日志
前一篇文章介绍了log4j2 扩展日志级别,支持将系统日志与业务处理日志拆分,现在介绍一下通过日志级别将系统日志与业务日志分类,要达到这个目的很容易,只需要配置一下log4j的xml文件: <? ...
- 前端知识学习——html
<!-- Html,CSS,JS 三者的关系 ==> 人,衣服,动作. 以下展示 html 常用基本编码 --><!-- Html 在PyCharm中新建html文件默认给出的 ...
- 并发策略-CAS算法
对于并发控制而言,我们平时用的锁(synchronized,Lock)是一种悲观的策略.它总是假设每一次临界区操作会产生冲突,因此,必须对每次操作都小心翼翼.如果多个线程同时访问临界区资源,就宁可牺牲 ...
- MVC如何在路由器(RouteConfig)定义后缀.html
一.配置文件web.config添加一下设置 <system.webServer> <modules runAllManagedModulesForAllRequests=" ...
- 你不知道的 JavaScript 基础细节
语法部分 type 属性: 默认的 type 就是 javascript, 所以不必显式指定 type 为 javascript javascript 不强制在每个语句结尾加 “:” , javasc ...
- php 快速上手
基本语法 PHP 标记 php 以 <?php 和 ?>(可以选用, 推荐, 防止输出空白) 作为开始和结束标记. 打印 php 有 echo(多参数) 和 print(单参数) 可以用于 ...
- luogu P1182 数列分段Section II
题目描述 对于给定的一个长度为N的正整数数列A[i],现要将其分成M(M≤N)段,并要求每段连续,且每段和的最大值最小. 关于最大值最小: 例如一数列4 2 4 5 1要分成3段 将其如下分段: [4 ...
- Creating and Flashing UBIFS with MTD Utils
转:http://wiki.atlas-embedded.com/index.php?title=Creating_and_Flashing_UBIFS_with_MTD_Utils Contents ...