ES5 规范之严格模式详解

前言

ECMAScript 5最早引入了“严格模式”(strict mode)的概念。通过严格模式,可以在函数内部存在的错误,及时捕获一些可能导致编程错误的ECMAScript 行为。

理解严格模式的规则非常重要,ECMAScript的下一个版本将以严格模式为基础制定。支持严格模式的浏览器包括IE10+、Firefox 4+、Safari 5.1+和Chrome。

严格模式的目的

  • 消除 JavaScript 语法的一些不合理、不严谨之处,减少一些怪异行为
  • 消除代码运行的一些不安全之处,保证代码运行的安全
  • 提高编译器效率,增加运行速度
  • 为未来新版本的 JavaScript 做好铺垫

严格模式的开启

想要开启严格模式,直接在作用域开始的位置写上字符串 "use strict";

在全局模式下开启:

"use strict";

在局部模式下开启(在函数中打开严格模式):

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

严格模式的规则(执行限制)

1、变量 var:消除了伪全局变量

在严格模式下,什么时候创建变量以及怎么创建变量都是有限制的。不允许意外创建全局变量(就是严格模式下消除了伪全局变量)

// 注意:(function(){})()是一个匿名函数
        (function () { 
            a = 10;
            console.log(a);
        })(); // 结果:10
        
        (function () {
            "use strict";
            a = 10;
        })();  // 报错:ReferenceError: a is not defined

在非严格模式下,即使变量 a 前面没有 var 关键字,即使没有将它定义为某个全局对象的属性,也能将 a 创建为伪全局变量使用。

在严格模式下,如果给一个没有声明的变量赋值,那代码在执行的适合就会抛出 ReferenceError(引用错误)

2、对函数参数的要求:更为严格

严格模式要求命名函数的参数必须唯一

        (function () { 
            function foo(a ,a, b){
                 console.log(a,b);
             }
            foo(1,2,3)
         })();  // 结果: 2 3
         
         (function () { 
             "use strict";
             function foo(a ,a, b){
                 console.log(a,b);
             }
             foo(1,2,3)
         })(); // 报错:SyntaxError: Duplicate parameter name not allowed in this context

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

在严格模式下,上面函数参数的不规范会抛出 SyntaxError(对象代表尝试解析语法上不合法的代码的错误)

3、实参形参 和 arguments 的分离

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

       // arguments 会受到 形参赋值的影响;
        (function(){
            function foo(a,b){
                a = 20;
                console.log(a,b);  // 20  2
                console.log(arguments[0],arguments[1]); // 20  2
            }
            foo(1,2)
        })(); 
        
        (function(){
            "use strict";
            // 形参 和 arguments 之间的区别;
            // 形参是变量可以随意赋值;
            // arguments 就是对应实参的关键字获取所有的实参,进行使用,不会被改变;
            function foo(a,b){
                a = 20;
                console.log(a,b);  // 20 , 2 
                console.log(arguments[0],arguments[1]);  //  1 , 2  
            }
            foo(1,2)
        })();

以上代码中,函数 foo() 传入两个参数 a,b 。调用这个函数时传入了两个参数“1,2”,这个值赋个了对应的变量。而在函数内部,a 的值被修改为“20”。

在非严格模式下,这个修改的值也会改变 arguments[0] 的值。

但在严格模式下,arguments[0] 的值仍然是传入的值。

4、arguments 的严格使用,部分功能禁用了

淘汰了 arguments.callee 和 arguments.caller。在非严格模式下,这两个属性一个是引用函数本身,一个是引用调用函数。而在严格模式下,这两个属性都被禁用了。

        (function(){
            function foo(){
               // arguments.callee 指向了当前的函数本身;
               console.log(arguments.callee);
            }
            foo()
        })();  // 结果 ƒ foo(){console.log(arguments.callee);}
        
        (function(){
            "use strict";
            function foo(){
               // 禁用掉了大部分arguments的属性;
               console.log(arguments.callee)
            }
            foo()
        })();  // 报错:TypeError: 'caller', 'callee', and 'arguments' properties may not be accessed on strict mode functions or the arguments

类似的,尝试读写函数的 caller 属性也会抛出 TypeError(类型错误)。

5、严格模式之中对 this 的严格(抑制this)

JavaScript 中一个最大的安全问题,也是最容易让人迷惑的地方就是在某些情况下如何抑制 this 的值。在非严格模式下,null 或 undefined 值会被转换为全局对象window。而在严格模式下,函数的 this 值始终是指定的值,无论指定的是什么值。

        (function(){
            function foo(){
                // this 指向 window;
               console.log(this);
            }
            foo()
        })(); // 结果:Window 
        
        (function(){
            "use strict";
            function foo(){
                // 禁用指向了 window 的 this,让 this 指向 undefined
               console.log(this);
            }
            foo()
        })();  // 结果:undefined

在以后的编程之中,this 最没有用的指向就是 全局对象window。

6、禁用 with(){} 语句

在非严格模式下的 with 语句能够改变解析标识符的路径,但在严格模式下,with 被抛弃了。因此,在严格模式下使用 with 会导致语法错误。

        (function(){    
            with(Math){
                // 可以省略对象前置;
                console.log(random()); // => Math.random()
                console.log(PI);  // => Math.PI
            }
        })();  
        
        // 严格模式之下禁用 with(){}
        (function(){
            "use strict";
            with(Math){
                console.log(random());
                console.log(PI);
            }
        })();  // 报错:Uncaught SyntaxError: Strict mode code may not include a with statement

在非严格模式下,with可以改变作用域链,他可以让他里面的代码的作用域链的最顶端变成with括号里面的这个对象,作用域链是经过很复杂的情况生成的结构,作用域链改了之后,系统内核会消耗大量的效率去更改作用域链,是会把程序变得非常慢的。所以ES5 的严格模式为了提高效率,禁用 with 语句

7、在严格模式之中被禁用的进制:不允许使用八进制

以 0 开头的八进制字面量过去经常会导致很多错误,在严格模式下,八进制字面量已经成为无效的语法了。

        (function(){
            // 0 开头就是八进制的标志;
            console.log(012);
        })(); // 结果: 10  (自动转化成十进制)
        
        (function(){
            "use strict"
            console.log(012)
        })(); //报错:Uncaught SyntaxError: Octal literals are not allowed in strict mode.
补充:ES5 也修改了严格模式下的 parseInt() 的行为。如今八进制的字面量在严格模式下会被当作以 0 开头的十进制字面量。例如:
        (function(){
            var value = 012;
            console.log(value);
         })(); // 结果: 10 
         
         (function(){
            "use strict";
            var value = parseInt("012");
            console.log(value); 
         })(); // 结果: 12 
         

如有遗漏之处欢迎评论区留言。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,752评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,100评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,244评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,099评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,210评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,307评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,346评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,133评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,546评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,849评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,019评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,702评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,331评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,030评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,260评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,871评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,898评论 2 351

推荐阅读更多精彩内容

  • 官网 中文版本 好的网站 Content-type: text/htmlBASH Section: User ...
    不排版阅读 4,380评论 0 5
  • 数值 判断NaN更可靠的方法是,利用NaN为唯一不等于自身的值的这个特点,进行判断。 isFinite方法返回一个...
    guyigg阅读 1,155评论 0 2
  • 一、概述: JavaScript由于语法的不严谨,一直所被人诟病。于是在ECMAscript 5添加了第二种运行模...
    不洗头的野人阅读 734评论 2 2
  • 1 概述 1.1函数的声明 JavaScript 有三种声明函数的方法。 (1)function 命令 funct...
    徵羽kid阅读 411评论 0 1
  • 函数和对象 1、函数 1.1 函数概述 函数对于任何一门语言来说都是核心的概念。通过函数可以封装任意多条语句,而且...
    道无虚阅读 4,551评论 0 5