JavaScript - 严格模式 - 作用域


今天先从一个小案例说起,面向对象的内容已经过去一半,剩下的内容也至关重要 ! 希望帮到你的朋友, 能持续关注更新 !

需求 : 图书管理操作

  • 示例代码 :
var list = [
        {name:"专业主义",author:"大前研一"},
        {name:"红楼梦",author:"曹公子"},
        {name:"什么是批判",author:"福柯"},
        {name:"十万个为什么?",author:"作者不详"}
    ];
    //01 提供构造函数
    function BookListManager(){
        this.bookList = null;
    }

    //02 设置原型对象
    BookListManager.prototype = {
        init:function(arrayList){
            this.bookList = arrayList || [];
        },
        getBookByName:function (name) {
        //01 能够找到该对象,返回
        for (var i = 0; i < this.bookList.length; i++)
        {
            var obj = this.bookList[i];
            if (obj.name == name)
            {
                return obj;
            }
        }

        //02 找不到这个对象,返回null
        return null;
    },
        addBook:function (bookObj){
        this.bookList.push(bookObj);
    },
        updateBook:function (name,author){
        var book = this.getBookByName(name);   //在原型方法中要调用其它的原型方法需要使用this
        book.author = author;
    },
        removeBook:function (name){
        var book = this.getBookByName(name);
        var index = this.bookList.indexOf(book); //返回指定对象的索引,如果没有那么就返回-1
        if(index == -1)
        {
            throw "要删除的数据不存在!";
        }
        //删除对象
        this.bookList.splice(index,1);
    }
    }

    //03 创建对象
    var p1 = new BookListManager();
    var p2 = new BookListManager();

    console.log(p1);
    console.log(p2);
    p1.init([{name:"城堡",author:"卡夫卡"}]);
    p2.init([{name:"一分钟学会H5开发",author:"乌拉乌拉"}]);

    console.log(p1);
    console.log(p2);
</script>
  • 总结 : 这个小Demo就是面向对象编程的示例,自习看代码的话会不会觉得封装性很好,以后可以直接拿到项目中应用,而且不用担心和别的同事的代码发生冲突.这就是面向对象编程的好处,代码看起来逼格也很高

严格模式

说明

  • 在js开发中可以选择严格模式或者是非严格模式来进行开发,默认是非严格模式的
  • 区别:在严格模式下,会做更严格的检查,一以前在非严格模式下不会报错或者是可以使用的一些语句在严格模式下使用会报错或者是抛出异常。
  • 严格模式它本身做了向后兼容:如果是支持严格模式的引擎,在检测到这个字符串的时候,就会在作用域中开启严格模式,如果是不支持严格模式的引擎,那么就直接忽略处理
  • 建议:推荐以后全部使用严格模式。
  • 语法 : "use strict" 在提供字符串设置严格模式的时候,必须要把字符串写在当前作用域的最顶端

** "use strict" 注意点 :
1.在书写字符串的使用双引号和单引号都可以
2.字符串后面的分号可以省略
3.所有的字符必须全部小写
4.这个字符串只能拥有10个字符**

使用注意点

  • 所有的变量都必须要使用var声明
  • 不能使用delete关键字来删除全局变量
  • 函数的形参列表中不能出现同名的参数
  • 在对象内部不能出现相同的属性
  • 禁止使用八进制
  • 禁止使用with语句
  • 禁止使用eval 和arguments作为标识符( 标识符就是不要这让两个参数作为变量使用 不然会报错 !)
  • 在函数调用的时候,修正了函数内部this的指向
  • 示例代码 :
<script>
  "use strict";
   var obj1 = {
        name:"张三",
        showName:function(){
            console.log(this);
        }
    }

    obj1.showName();   //obj1            this--->obj1
    var showName = obj1.showName;
    showName();         //this丢失的问题  this--->window(非严格)|undefined(严格)

    obj1.showName.call(null);  
    //在严格模式下,函数内部的this绑定谁由第一个参数决定,如果没有传递那么是undefined,如果传入的值是null,那么绑定的就是null
</script>
  • 禁止使用callee | caller
  • arguments在函数内部的表现和非严格模式不同

1.arguments 接收实参的值
2.值类型&&引用类型的数据作为函数的参数传递,在非严格模式下,arguments和形参共享一份数据
3.重新设置了形参的值,arguments的值也会发生改变,在严格模式下,arguments和形参是独立的,他们之间没有关系,修改了形参的值对arguments没有影响的

  • 示例代码 :
<script>
  "use strict";
    function funcName(paramStr){
        console.log(paramStr);
        console.log(arguments[0]);

        //重新设置形参的值
        paramStr = "测试的字符串";
        console.log(arguments[0]);
    }
    var str = "demoStr";
    funcName(str);

    function funcName02(obj)
    {
        console.log(obj);
        console.log(arguments[0]);

        //重新设置obj的值
        obj = {age:20};
        console.log(obj);
        console.log(arguments[0]);
    }
    var obj = {
        name:"张三"
    };

    funcName02(obj);
</script>

严格模式的作用域

写在当前作用域的最顶端
作用范围
  • script标签 只对当前的标签有影响,仅仅设置当前的标签为严格模式
  • 函数内 那么只对当前的函数作用域有效
  • 示例代码 :
<script>
     // "use strict";     位置01 对f1和f2都有效
    function f1(){
     // "use strict";     位置02 对f1有效f2无效

        //a = 10;
    }
    function f2(){
    // "use strict"; 位置03  对f2有效f1无效
        b = 20;
    }
    f1();
    f2();
</script>

作用域

某个变量有( 起 ) 作用的范围
块级作用域, 在别的语言里有块级作用域, 但是在js中没有块级作用域
js中的作用域
  • script构成了全局作用域
  • 在js中函数是唯一一个可以创建作用域的对象
词法作用域 - 动态作用域
  • 词法作用域:在变量声明的时候,它的作用域就已经确定了
  • 动态作用域:在程序运行的时候,由程序的当前上下文(执行环境)决定的
  • js属于词法作用域
词法作用域的访问规则:
  • 先在当前作用域中查找,如果找到就直接使用,如果没有找到,那么就到上一级作用域中查找,如果还没有找到那么就重复这个查找的过程,直到全局作用域

变量和函数提升

js的执行顺序
  • 预解析阶段 变量和函数的提升(声明)
  • 具体的执行阶段
  • 变量和函数的提升
  • 示例代码 :
<script>
    console.log(a);   //undefined ?
    var a = 10;

    f1();
    function f1(){
        console.log("f1");
    }
</script>
  • 变量和变量同名的情况:后面的变量会把前面的变量覆盖
  • 示例代码 :
<script>
//    var n1 = "n1";
//    console.log(n1);                //n1
//
//    function test(){
//        console.log(n1);
//    }
//
//    test();                         //n1
//    var n1 = "new n1";
//    console.log(n1);                //new n1
//    test();                         //new n1


//      模拟变量和函数的提升
        var n1;
        function test(){
            console.log(n1);
        }
        var n1;
    n1 = "n1";
    console.log(n1);                //n1

    test();                         //n1
    n1= "new n1";
    console.log(n1);                //new n1
    test();                         //new n1
</script>
  • 函数和函数同名的情况
  • 示例代码 :
<script>
    f1();                   //10 ? 20
    function f1(){
        console.log(10);
    }

    f1();                   //20
    function f1(){
        console.log(20);
    }
    f1();                   //20
</script>
  • 函数和变量同名的情况:可以理解为函数声明提升而变量的声明不提升(忽略)
  • 示例代码 :
<script>
    console.log(demo);                      //
    var demo = "我是字符串";
    function demo(){
        console.log("我是函数");
    }

    console.log(demo);                      //

    //......
    var demo;
    function demo(){
        console.log("我是函数");
    }

    console.log(demo);                      //函数
    demo = "我是字符串";
    console.log(demo);                      //我是字符串


</script>
<script>
    console.log(test);                      //函数
    function test(){
        console.log("我是函数");
    }
    var test = "我是帅哥";
    console.log(test);                      //

    ///////变量声明(和函数同名的)不会提升
    function test(){
        console.log("我是函数");
    }
    console.log(test);                      //
    var test = "我是帅哥";
    console.log(test);
</script>
注意 : 如果是函数的表达式,那么在做函数声明提升的时候,仅仅只会把var 变量的名称(函数)提升到当前作用域的最顶端

作用域链

  • js中函数可以创建作用域
  • js中的函数中可以声明函数
  • 函数内部的函数中又可以声明函数
  • 以上,会形成一个链式的结构,这个是作用域链
  • 示例代码 :
<script>

    var a = 10;
    function f1(){

        function f2(){

            var d = "demoD";
        }
    }

    function f3(){
        function f4(){

        }

        var b = "demoB";

        function f5(){

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

推荐阅读更多精彩内容