欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

JavaScript 作用域的误区

程序员文章站 2022-06-24 14:58:30
了解JavaScript的同学可能知道,JavaScript语言由于设计原因,导致语言本身存在很多先天性的不足,当然这并非设计者有意的,js语言最初是被设计来作为网页交互的脚本语言,依照现有的js语法来看,其最初的需求已经完全能够满足。互联网的发展极大的提高了对web端的要求,不仅仅要求简单的页面交 ......

了解javascript的同学可能知道,javascript语言由于设计原因,导致语言本身存在很多先天性的不足,当然这并非设计者有意的,js语言最初是被设计来作为网页交互的脚本语言,依照现有的js语法来看,其最初的需求已经完全能够满足。互联网的发展极大的提高了对web端的要求,不仅仅要求简单的页面交互,已经趋于app的概念,这才暴漏出javascript语言设计上的种种问题。对于开发者来说,理清“坑点”并在实际开发中正确的规避才是重点。

1、任何js代码都有自己的作用域

  es6之前,使用var声明变量的时候,很容易被忽略的一点是,var声明的变量的作用域;

  1)在函数中var声明的变量,其作用域是在函数体内部,即函数外部是访问不到的:

<script>
        function foo() {
            var myname = "foo_name";
            console.log(myname);//foo_name
        }
        foo();
        console.log(myname);//error
</script>

  这是很容易理解的,因为javascript本身有函数作用域。

  2)不在函数中var声明的变量:

<script>
        var myname_1 = "myname_1";
        {
            var myname_2 = "myname_2";
        }
        console.log(myname_1);//myname_1
        console.log(myname_2);//myname_2,不会error
</script>

  这也很容易理解,因为javascript中没有块级作用域,在{}中声明的变量和不使用{}没有区别。

  3)这里很容易很忽略的一点是,以上js代码都有作用域,就是全局作用域window,所有声明的变量都有自己的作用域,只不过常见的是作用域是全局作用域即window,很容易被忽略掉,导致没有作用域的认知。有时候这点误区会导致验证的错误。

  新建html页面html1.html,首先看看如何验证以上的说法,

<script>
        function showname () {
            console.log("这里是showname");
        }
        window.showname();//这里可以验证shownname的作用域在window
</script>

  以上代码很容易验证了showname函数的作用域是window。

  新建页面html2.html,并写入一下js:

<script>
        function shownametoo () {
            console.log("这里是shownametoo");
        }
        window.shownametoo();
</script>

  现在在html1页面中使用iframe引用html2页面,在html1中添加js代码:

<script>
        function showname () {
            console.log("这里是showname");
        }
        window.showname();
        var iframe = document.queryselector("#iframe");
        iframe.src = "html2.html";
        window.iframe.contentwindow.shownametoo();
</script>

  在html2.html中添加如下代码:

<script>
        function shownametoo () {
            console.log("这里是shownametoo");
        }
        window.shownametoo();
        window.parent.showname();
</script>

  在页面上打html1(通过服务器打开,不能直接打开文件),可以看到有趣的现象。

  总结:之所以讲这个例子,是因为不少新加入前端的同学都会有一个误区,就是js中只有函数作用域(忽略全局作用域,因为使用的很少)。其实这个观点很重要,在多模块开发中,一个模块就相当于一个页面的windiw作用域。