javascript的做用域

毕业了最近在找工做,面试一家公司遇到了许多javascript的题,由于只是之前稍微了解了一点,遇到了一些之前没接触的东西。许多的面试题里面会有读程序的题,当时不会作回来一搜发现这些题都是在考js的做用域。javascript

1、JavaScript做用域html

JavaScript变量实际上只有两种做用域,全局变量和函数的内部变量。在函数内部任何一个地方定义的变量(var scope)其做用域都是整个函数体。
全局变量:指的是window对象下的对象属性。
做用域划分:基于上下文,以函数进行划分的,而不是由块划分的。
强调两点:
1. 在同一做用域中,JavaScript是容许变量的重复定义,而且后一个定义将覆盖前一个定义。
2. 函数内部若是不加关键字var而定义的变量,默认为全局变量。java

var scope="global"; 
function t(){ 
  console.log(scope); //"global" 
  scope="local"
  console.log(scope); //"local" 
} 
t(); 
console.log(scope); //"local" 
 
 
 
 
var scope="global"; 
function t(){ 
  console.log(scope); //"undefined" 
  var scope="local"
  console.log(scope); //"local" 
} 
t(); 
console.log(scope); //"global"

在变量解析过程当中首先查找局部的做用域,而后查找上层做用域。在第一段代码的函数当中没有定义变量scope,因而查找上层做用域(全局做用域),进而进行输出其值。可是在第二段代码的函数内定义了变量scope(不管是在console以后仍是以前定义变量,都认为在此做用域拥有变量scope),因而再也不向上层的做用域进行查找,直接输出scope。可是不幸的是此时的局部变量i并无赋值,因此输出的是undefined。面试

//因此根据函数做用域的意思,能够将上述第二段代码重写以下: 
var scope="global"; 
function t(){ 
  var scope; 
  console.log(scope); //“local”
  scope="local"
  console.log(scope); //“local”
} 
t();

因为函数做用域的特性,局部变量在整个函数体始终是由定义的,咱们能够将变量声明”提早“到函数体顶部。数组

var b; //第1步 
function fun(){  
  b = "change";  
}  
alert(b);//输出undefined,因为第1步只定义未赋值 
  
  
var b; //第1步 
function fun(){  
  b = "change";  
} 
fun(); //调用上述函数 
alert(b); //输出change

当使用var声明一个变量时,建立的这个属性是不可配置的,也就是说没法经过delete运算符删除。
2、做用域实例闭包

<html> 
<head> 
  <script type="text/javascript"> 
    function buttonInit(){ 
      for(var i=1;i<4;i++){ 
        var b=document.getElementById("button"+i); 
        b.addEventListener("click",function(){ alert("Button"+i);},false); 
      } 
    } 
    window.onload=buttonInit; 
  </script> 
</head> 
<body> 
  <button id="button1">Button1</button> 
  <button id="button2">Button2</button> 
  <button id="button3">Button3</button> 
</body> 
</html>

依次弹出button1,button2,button3,当注册事件结束后,i的值为4,当点击按钮时,事件函数即function(){ alert("Button"+i);}这个匿名函数中没有i,根据做用域链,因此到buttonInit函数中找,此时i的值为4,因此弹出”button4“。函数

3、利用js闭包实现循环绑定事件this

<html> 
<head> 
  <title>闭包</title> 
</head> 
<body> 
  <ul id="list"> 
    <li>第1条记录</li> 
    <li>第2条记录</li> 
    <li>第3条记录</li> 
    <li>第4条记录</li> 
    <li>第5条记录</li> 
    <li>第6条记录</li> 
  </ul> 
  <script type="text/javascript"> 
    function tt(nob) { 
      this.clickFunc = function() { 
        alert("这是第" + (nob + 1) + "记录"); 
      } 
    } 
    var list_obj = document.getElementById("list").getElementsByTagName("li"); //获取list下面的全部li的对象数组 
    for (var i = 0; i<= list_obj.length; i++){ 
      console.log(list_obj[i]) 
      list_obj[i].onmousemove = function(){ 
        this.style.backgroundColor = "#cdcdcd"; 
      } 
      list_obj[i].onmouseout = function() { 
        this.style.backgroundColor = "#FFFFFF"; 
      } 
      //list_obj[i].onclick = function() { 
      // alert("这是第" + i + "记录"); //不能正常获取 alert出来的都是:“这是第6记录” 
      //} 
      var col = new tt(i); //调用tt函数 
      list_obj[i].onclick = col.clickFunc; //执行clickFunc函数 
    } 
  </script> 
</body> 
</html>

这里的for循环中一开始就会循环完毕,若是不是经过tt()传入的nob记录的话,就不能正常获取alert出来的数。code

相关文章
相关标签/搜索