返回介绍

作用域问题

发布于 2024-06-02 22:07:05 字数 5390 浏览 0 评论 0 收藏 0

JavaScript语言的作用域仅存在于函数范围中。这是必须要牢记的一点,还有一点重要的就是作用域的提升规则。

作用域问题

JS最容易出现混淆的就是作用域的情况。传统的类C语言,它们的作用域是block-level scope,块级作用域, 花括号就是一个作用域。但是对于JavaScript而言,它的作用域是function-level scope,比如if条件语句,就不算一个独立的作用域:

var x = 1;
console.log(x); // 1
if (true) {
    var x = 2;
    console.log(x); // 2
}
console.log(x); // 2

在JavaScript中,如果我们需要实现block-level scope,我们也有一种变通的方式,那就是通过自执行函数创建临时作用域:

function foo() {
    var x = 1;
    if (x) {
        (function () {
            var x = 2;
            // some other code
        }());
    }
    // x is still 1.
}

作用域提升

变量被提升

对JavaScript解释器而言,所有的函数和变量声明都会被提升到最前面, 并且变量声明永远在前面,赋值在声明过程之后。比如:

var x = 10;
function x(){};
console.log(x); // 10

实际上被解释为:

var x;
function x(){};
x = 10;
console.log(x); // 10

函数被提升

函数的声明方式主要由两种:声明式和变量式。

声明式会自动将声明放在前面并且执行赋值过程。而变量式则是先将声明提升,然后到赋值处再执行赋值。比如:

function test() {
    foo(); // TypeError "foo is not a function"
    bar(); // "this will run!"
    var foo = function () { // function expression assigned to local variable 'foo'
        alert("this won't run!");
    }
    function bar() { // function declaration, given the name 'bar'
        alert("this will run!");
    }
}
test();

实际上等价于:

function test() {
    var foo;
    var bar;
    bar = function () { // function declaration, given the name 'bar'
        alert("this will run!");
    }

    foo(); // TypeError "foo is not a function"
    bar(); // "this will run!"

    foo = function () { // function expression assigned to local variable 'foo'
        alert("this won't run!");
    }
}
test();

主要注意的地方:带有命名的函数变量式声明,是不会提升到作用域范围内的,比如:

var baz = function spam() {};
baz(); // vaild
spam(); // ReferenceError "spam is not defined"

实战经验

任何时候,请使用var声明变量, 并放置在作用域的顶端.

工具推荐JSLint之类,帮助你验证语法的规范。

参考资料

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据
    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文