我們的志工尚未將此文章翻譯為 正體中文 (繁體) 版本。加入我們,幫忙翻譯!
ECMAScript 5 introduced strict mode which is now implemented in all major browsers (including IE10). While making web browsers interpret code as strict is easy (just add "use strict";
at the top of your source code), transitioning an existing code base to strict mode requires a bit more work.
This article aims at providing guidance for developers.
Gradual transition
Strict mode has been designed so that the transition to it can be made gradually. It is possible to change each file individually and even to transition code to strict mode down to the function granularity.
Differences from non-strict to strict
Syntax errors
When adding "use strict";
, the following cases will throw a SyntaxError
before the script is executing:
- Octal syntax
var n = 023;
with
statement- Using
delete
on a variable namedelete myVariable
; - Using
eval
orarguments
as variable or function argument name - Using one of the newly reserved keywords (in prevision for ECMAScript 6):
implements
,interface
,let
,package
,private
,protected
,public
,static
, andyield
- Declaring function in blocks
if(a<b){ function f(){} }
- Obvious errors
- Declaring twice the same name for a property name in an object literal
{a: 1, b: 3, a: 7}
This is no longer the case in ECMAScript 6 (bug 1041128). - Declaring two function parameters with the same name
function f(a, b, b){}
- Declaring twice the same name for a property name in an object literal
These errors are good, because they reveal plain errors or bad practices. They occur before the code is running.
New runtime errors
JavaScript used to silently fail in contexts where what was done was an error. Strict mode throws in such cases. If your code base contains such cases, testing will be necessary to be sure nothing is broken. Once again, it can happen at the function granularity level.
Setting a value to an undeclared variable
function f(x){ "use strict"; var a = 12; b = a + x*35; // error! } f(42);
This used to change a value on the global object which is rarely the expected effect. If you really want to set a value to the global object, pass it as an argument and explicitly assign it as a property:
var global = this; // in the top-level context, "this" always // refers to the global object function f(x){ "use strict"; var a = 12; global.b = a + x*35; } f(42);
Trying to delete a non-configurable property
"use strict"; delete Object.prototype; // error!
In non-strict, this would silently fail, in contradiction with the user expectation.
Poisoned arguments and function properties
Accessing arguments.callee
, arguments.caller
, anyFunction.caller
, or anyFunction.arguments
throws an error in strict mode. The only legitimate use case would be to reuse a function as in:
// example taken from vanillajs: https://vanilla-js.com/ var s = document.getElementById('thing').style; s.opacity = 1; (function(){ if((s.opacity-=.1) < 0) s.display="none"; else setTimeout(arguments.callee, 40); })();
which can be rewritten as:
"use strict"; var s = document.getElementById('thing').style; s.opacity = 1; (function fadeOut(){ // name the function if((s.opacity-=.1) < 0) s.display="none"; else setTimeout(fadeOut, 40); // use the name of the function })();
Semantic differences
These differences are very subtle differences. It's possible that a test suite doesn't catch this kind of subtle difference. Careful review of your code base will probably be necessary to be sure these differences don't affect the semantics of your code. Fortunately, this careful review can be done gradually down the function granularity.
this
in function calls
In function calls like f()
, the this
value was the global object. In strict mode, it is now undefined
. When a function was called with call
or apply
, if the value was a primitive value, this one was boxed into an object (or the global object for undefined
and null
). In strict mode, the value is passed directly without conversion or replacement.
arguments
doesn't alias named function arguments
In non-strict mode, modifying a value in the arguments
object modifies the corresponding named argument. This made optimizations complicated for JavaScript engine and made code harder to read/understand. In strict mode, the arguments
object is created and initialized with the same values than the named arguments, but changes to either the arguments
object or the named arguments aren't reflected in one another.
Change to eval
In strict mode code, eval
doesn't create a new variable in the scope from which it was called. Also, of course, in strict mode, the string is evaluated with strict mode rules. Thorough testing will need to be performed to make sure nothing breaks. Not using eval if you don't really need it may be another pragmatic solution.
Strictness-neutral code
A potential "downside" of moving strict code to strict mode is that the semantics may be different in legacy browsers which do not implement strict mode. In some rare occasions (like bad concatenation or minification), your code also may not run in the mode you wrote and tested it in. Here are the rules to make your code strictness-neutral:
- Write your code as strict and make sure no strict-only errors (from the above "New runtime errors" section) are thrown.
- Stay away from semantic differences
eval
: use it only if you know what you're doingarguments
: always access function arguments via their name or perform a copy of the arguments object using:
var args = Array.prototype.slice.call(arguments)
as the first line of your functionthis
: only usethis
when it refers to an object you created.