严格模式(Strict Mode)是ECMAScript 5的新特性,它允许你把整个程序,或者某个函数,放置在“严格”的操作语境中。这种严格的语境会防止某些特定的操作并抛出更多的异常。
虽然ECMAScript 5对ECMAScript 3是向下兼容的,但是在严格模式下,所有在ECMAScript 3中不赞成使用的特性都被禁用(或抛出错误)而不是兼容。
支持严格模式的浏览器:Firefox 5+、IE9、Chrome
启用严格模式有以下好处:
- 捕获一些编程错误,并抛出异常。
- 阻止进行一些相对“不安全”的操作(例如访问全局变量),抛出异常。
- 禁用一些让人迷惑的特性。
关于严格模式的大多数信息都可以在http://es5.github.com/#x4.3.6中找到。
-
如何启用严格模式
在程序的开头添加这条语句即可对整段脚本启用严格模式:
'use strict';
也可以只在函数的内部启用严格模式,这样不会影响到外部:
function imStrict() { 'use strict'; // ... your code ... }
启用严格模式的语句只是一段普通的字符串“use strict”,没有任何新语法。这意味着不会对就旧式浏览器造成任何负面影响。
在函数内部启用严格模式的一个实际应用是,把整个Javascript类库定义在严格模式的函数内部,这样就可以不影响外部的代码:
// Non-strict code...
(function(){ "use strict";
// Define your library strictly... })();
// Non-strict code...
启用了严格模式后,不赞成的特性将的控制台中提示错误。
开启严格模式后,以下操作将会提示错误:
-
变量和属性
对未定义变量的赋值将会失败,而不是把这个变量作为全局变量。
写入一个可写特性为false的属性,删除一个可配置特性为false的属性,或者添加一个可扩展特性为false的属性,都会导致错误(这些特性是预先商定好的)。过去,这些操作不会抛出异常,仅仅是静默地失败。
对变量、函数或者函数参数执行delete操作会导致错误。
var foo = 'test'; function test() { }
delete foo; // Error delete test; // Error
function test2(arg) { delete arg; // Error }
在一个对象容器内部定义相同属性会导致异常的抛出:
// Error { foo: true, foo: false }
8进制数字表示被制止, 010 不再表示 8,而是提示不支持八进制字面量的语法错误
eval
道格拉斯曾说:“Eval是魔鬼”。Eval是非常灵活的条件元素,但是它也是一个大问题。从开始用它,如果广泛的使用,你的规则将变得不可读。同样,因为你可以在Eval中做几乎任何事情,引擎无法对它做任何假设,因此也很难提供任何优化。因此,任何对“eval”这个名字的使用(主要意图是把eval函数指向一个变量或者是对象的属性)都是禁止的。
// All generate errors... obj.eval = ... obj.foo = eval; var eval = ...; for ( var eval in ... ) {} function eval(){} function test(eval){} function(eval){} new Function("eval")
另外,通过eval声明新变量也会无效:
eval("var a = false;"); print( typeof a ); // undefined
-
函数
重写arguments对象会导致错误:
arguments = [...]; // not allowed
定义同名参数(function(foo, foo) { })会导致错误。
对arguments.caller和arguments.callee的访问会抛出异常。因此,任何需要用到的匿名函数都必须先命名,例如:
setTimeout(function later(){ // do stuff... setTimeout( later, 1000 ); }, 1000 );
函数的arguments、caller和callee属性不再存在,定义它们的操作也是禁止的。
function test() { } test.caller = 'caller'; // Error
最后,一个长期存在(且非常讨厌)的BUG已经被解决了:当使用null或者undefined作为Function.prototype.call或 Function.prototype.apply方法的第一个参数时,函数内部的this将会指向全局对象。而严格模式将会阻止其执行并抛出异常:
(function(){ ... }).call(null); // Exception
函数声明只能在top level 或者直接出现在其他函数中
下面这段代码将输出几?
var a = 1;
if(a == 1){
function fn(){alert(1);}
}else{
function fn(){alert(2);}
}
fn();
结果是2,在严格模式下不允许这样声明函数
with() { }
with() { }语句一直被认为是万恶之源,在严格模式下完全禁止使用该语句。