Javascript 严格模式详解(上)

  ECMAScript 5最早引进了“严格模式”(strict mode)的概念。通过严格模式,可以在函数内部选择进行较为严格的全局或局部的错误条件检测。使用严格模式的好处是可以提早知道代码中存在的错误,及时捕获一些可能导致变成错误的ECMAScript行为。支持严格模式的浏览器包括IE 10+,Firefox 4+,Safari 5.1+和Chrome。

 1.   选择使用

   要进入严格模式,可以使用严格模式的编译指示(pragma)

“use strict”;

 

  如果是在全局作用域中(函数外部)给出这个编译指示,则整个脚本都将使用严格模式。

  也可是只在函数中打开严格模式,就像下面这样:

function doSomething(){
       "use strict";
     //其他代码
}

 

  如果你没有控制页面中所有脚本的权利,则只需要在测试的特定函数中开启严格模式。

  2.   变量

  在严格模式下,什么时候创建变量以及怎么创建变量都是有限制的。首先,不允许意外创建全局变量。在非严格模式下,可以像那这样创建全局变量:

//未声明变量
//非严格模式:创建全局变量
//严格模式:抛出ReferenceError

message="Hello World!";

 

  即使message前面没有var关键字,即使没有将它定义为某个全局对象的属性,也能将message创建为全局变量。但在严格模式下,如果给一个没有声明的变量赋值,代码在执行时就会抛出ReferenceError。

  其次,不能对变量调用delete操作符。非严格模式允许这样操作,但会静默失败。而在严格模式下,删除变量也会导致错误。

//删除变量
//非严格模式:静默失败
//严格模式:抛出ReferenceError

var color="red";
delete color;

 

  严格模式下对变量名也有限制。特别的,不能使用implements,interface,let,package,private,protected,public,static和yield作为变量名。在严格模式下,用以上标识符作为变量名会导致语法错误。

 3.对象

  在严格模式下操作对象比在非严格模式下更容易导致错误。一般说来,非严格模式下会静默失败的情形,在严格模式下就会抛出错误。因此,在开发中使用严格模式会加大早发现错误的可能性。

  在下列情形下操作对象的属性会导致错误:

  • 为只读属性赋值会抛出TypeError;
  • 对不可配置的(nonconfigurable)的属性使用delete操作符会抛出TypeError;
  •  为不可扩展的(nonextensible)的对象添加属性会抛出TypeError。

   使用对象的另一个限制与通过对象字面量声明对象有关。在使用对象字面量时,属性名必须唯一。

例如:

//重名属性
//非严格模式:没有错误,以第二个属性为准
//严格模式:抛出语法错误

var person={
                name:"Nicholas",
                name:"Greg"
           };

 

   4.    函数

   首先,严格模式要求命名函数的参数必须唯一。以下面函数为例:

//重名参数
//非严格模式:没有错误,只能访问第二个参数
//严格模式:抛出语法错误

function sum(num,num){
                         //do sonmething
}

 

    在非严格模式下,这个函数声明不会抛出错误。通过参数名只能访问第二个参数,要访问第一个参数必须通过arguments对象。

    在严格模式下,arguments对象的行为也有所不同,在非严格模式下,修改命名参数的值也会反映到argument对象中,而严格模式下这两个值是完全独立的。例如:

//修改命名参数的值
//非严格模式:修改会反映到argument中
//严格模式:修改不会反映到argument中

function showValue(value){
       value="Foo";
       alert(value);             //"Foo"
       alert(argument[0]);  //非严格模式:"Foo"
                                    //严格模式:"Hi"
}

showValue("Hi");

   另一个变化是淘汰了arguments.callee和arguments.caller。在非严格模式下,这两个属性一个引用函数本身,一个引用调用函数。而在严格模式下,访问哪个属性都会抛出TypeError。例如:

//访问arguments.callee
//非严格模式:没有问题
//严格模式:抛出TypeError

function factorial(num){
     if(num<=1){
              return;
     }else{
              return num*arguments.callee(num-1)
     }
}

var result=factorial(5);

   类似的,尝试读写函数的caller属性,也会导致抛出TypeError。所以,对于上面的例子而言,访问factorial.caller也会抛出错误。

   与变量类似,严格模式对函数名也做出了限制,不允许用implements,interface,let,package,private,protected,public,static和yield作为函数名。

   对函数的最后一点限制,就是只能在脚本的顶级和在函数内部声明函数,也就是说,在if语句中声明函数会导致语法错误:

//在if语句中声明函数
//非严格模式:将函数提升到if语句外
//严格模式:抛出语法错误

if(true){
     function dosomething(){
               //.....
     }
}

  在非严格模式下,以上代码能在所有浏览器中运行,但在严格模式下会导致语法错误。

 

郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。