版本

no-loop-func

禁止包含循环语句内部不安全引用的函数声明

在循环中编写函数往往会导致错误,因为函数会围绕循环创建闭包。例如

for (var i = 0; i < 10; i++) {
    funcs[i] = function() {
        return i;
    };
}

在这种情况下,您期望在循环中创建的每个函数返回不同的数字。实际上,每个函数都返回 10,因为这是作用域中 `i` 的最后一个值。

`let` 或 `const` 可以缓解此问题。

for (let i = 0; i < 10; i++) {
    funcs[i] = function() {
        return i;
    };
}

在这种情况下,在循环中创建的每个函数都按预期返回不同的数字。

规则详情

引发此错误是为了突出可能无法按预期工作且可能表明对语言工作原理的误解的代码部分。如果您不修复此错误,您的代码可能会毫无问题地运行,但在某些情况下,它可能会出现意外行为。

此规则禁止循环内包含不安全引用(例如对来自外部作用域的已修改变量的引用)的任何函数。此规则忽略 IIFE,但不忽略异步或生成器函数。

此规则的错误代码示例

在 Playground 中打开
/*eslint no-loop-func: "error"*/

var i = 0;
while(i < 5) {
    var a = function() { return i; };
    a();

    i++;
}

var i = 0;
do {
    function a() { return i; };
    a();

    i++
} while (i < 5);

let foo = 0;
for (let i = 0; i < 10; ++i) {
    //Bad, `foo` is not in the loop-block's scope and `foo` is modified in/after the loop
    setTimeout(() => console.log(foo));
    foo += 1;
}

for (let i = 0; i < 10; ++i) {
    //Bad, `foo` is not in the loop-block's scope and `foo` is modified in/after the loop
    setTimeout(() => console.log(foo));
}
foo = 100;

var arr = [];

for (var i = 0; i < 5; i++) {
    arr.push((f => f)(() => i));
}

for (var i = 0; i < 5; i++) {
    arr.push((() => {
        return () => i;
    })());
}

for (var i = 0; i < 5; i++) {
    (function fun () {
        if (arr.includes(fun)) return i;
        else arr.push(fun);
    })();
}

此规则的正确代码示例

在 Playground 中打开
/*eslint no-loop-func: "error"*/

var a = function() {};

for (var i=10; i; i--) {
    a();
}

for (var i=10; i; i--) {
    var a = function() {}; // OK, no references to variables in the outer scopes.
    a();
}

for (let i=10; i; i--) {
    var a = function() { return i; }; // OK, all references are referring to block scoped variables in the loop.
    a();
}

var foo = 100;
for (let i=10; i; i--) {
    var a = function() { return foo; }; // OK, all references are referring to never modified variables.
    a();
}
//... no modifications of foo after this loop ...

var arr = [];

for (var i=10; i; i--) {
    (function() { return i; })();
}

for (var i = 0; i < 5; i++) {
    arr.push((f => f)((() => i)()));
}

for (var i = 0; i < 5; i++) {
    arr.push((() => {
        return (() => i)();
    })());
}

版本

此规则是在 ESLint v0.0.9 中引入的。

资源

更改语言