关于性能:JavaScript变量声明在循环外部或内部?

JavaScript variables declare outside or inside loop?

在AS3中,我认为你应该初始化循环外的所有变量以提高性能。 这也是JavaScript的情况吗? 哪个更好/更快/最佳实践?

1
2
3
4
5
6
var value = 0;

for (var i = 0; i < 100; i++)
{
    value = somearray[i];
}

要么

1
2
3
4
for (var i = 0 ; i < 100; i++)
{
    var value = somearray[i];
}


在JavaScript或ActionScript中,意义或性能完全没有区别。

var是解析器的指令,而不是在运行时执行的命令。如果特定标识符已在函数体(*)中的任何位置声明var一次或多次,则块中该标识符的所有使用都将引用局部变量。是否将value声明为循环内部,循环外部或两者都是var没有区别。

因此,你应该写出你认为最具可读性的。我不同意Crockford的说法,将所有变量放在函数顶部总是最好的。对于在一段代码中临时使用变量的情况,最好在该部分中声明var,因此该部分是独立的并且可以进行复制粘贴。否则,在重构期间将几行代码复制粘贴到一个新函数,而无需单独挑选和移动关联的var,并且您自己也是偶然的全局。

特别是:

1
2
3
4
5
for (var i; i<100; i++)
    do something;

for (var i; i<100; i++)
    do something else;

Crockford建议您删除第二个var(或同时删除var s并执行上面的var i;),jslint将为此向您发出警告。但IMO更容易保持两个var,将所有相关代码保持在一起,而不是在函数顶部有一个额外的,容易被遗忘的代码。

就个人而言,我倾向于将var声明为独立代码段中变量的第一个赋值,无论是否在同一函数的某个其他部分中使用相同的变量名称。对我来说,必须声明var是一个不受欢迎的JS wart(将变量默认为local更好);我不认为我有责任在JavaScript中复制[旧版本] ANSI C的限制。

(*:除嵌套函数体外)


理论上它不应该在JavaScript中有任何区别,因为语言没有块范围,只有函数范围。

我不确定性能参数,但Douglas Crockford仍然建议var语句应该是函数体中的第一个语句。引用JavaScript编程语言的代码约定:

JavaScript does not have block scope, so defining variables in blocks can confuse programmers who are experienced with other C family languages. Define all variables at the top of the function.

我认为他有一个观点,你可以在下面的例子中看到。声明函数顶部的变量不应该让读者误以为变量i保存在for循环块的范围内:

1
2
3
4
5
6
7
function myFunction() {
  var i;    // the scope of the variables is very clear

  for (i = 0; i < 10; i++) {
    // ...
  }
}


ECMA-/Javascript语言hoists任何在函数顶部声明的变量。那是因为这种语言确实有function scope,并且没有像许多其他类C语言那样具有block scope
这也称为lexical scope

如果你宣布类似的东西

1
2
3
4
var foo = function(){
    for(var i = 0; i < 10; i++){
    }
};

这得到hoisted

1
2
3
4
5
var foo = function(){
    var i;
    for(i = 0; i < 10; i++){
    }
}

所以它在性能上没有任何差别(但如果我在这里完全错了,请纠正我)。
不在函数顶部而不是在函数顶部声明变量的更好的论据是可读性。在for-loop中声明变量可能会导致错误的假设,即此变量只能在循环体中访问,这是完全错误的。事实上,您可以在当前范围内的任何位置访问该变量。


明年,所有浏览器都会有预编译代码的JS引擎,因此性能差异(通过一次又一次地解析相同的代码块以及执行赋值)应该可以忽略不计。

此外,除非必须,否则永远不要优化性能。第一次将变量保持在您需要的位置附近可以保持代码清洁。从消极方面来说,习惯于使用块范围的语言的人可能会感到困惑。


我只是在Chrome中做了一个简单的测试。在浏览器中尝试小提琴并查看结果

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
  var count = 100000000;
    var a = 0;
    console.log(new Date());

    for (var i=0; i<count; i++) {
      a = a + 1
    }

    console.log(new Date());

    var j;
    for (j=0; j<count; j++) {
      a = a + 1;
    }

    console.log(new Date());

    var j;
    for (j=0; j<count; j++) {
        var x;
        x = x + 1;
    }

    console.log(new Date());

结果是最后一次测试需要大约8秒,前两次测试只需要~2秒。非常重复,无论顺序如何。

所以,这证明了我应该总是在循环之外声明变量。好奇的情况是我在for()语句中声明i的第一个。这个看起来和我预先声明索引的第二个测试一样快。


现在我们在ES2015中有letconst的另一个考虑因素是,您现在可以将变量专门用于循环块。因此,除非你需要在循环外部使用相同的变量(或者如果每次迭代都依赖于在前一次迭代中对该变量所做的操作),那么最好这样做:

1
2
3
4
for (let i = 0; i < 100; i++) {
    let value = somearray[i];
    //do something with `value`
}


JavaScript是一种用C或C ++底部编写的语言,我不太确定它是哪一种。其目的之一是节省处理内存的重要性。
即使在C或C ++中,您也不必担心在循环内声明变量时是否会消耗大量资源。你为什么要在JavaScript中担心呢?


如果在for循环内部或外部声明变量,则没有区别。
下面是要测试的示例代码。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
function a() {
   console.log('Function a() starts');
   console.log(new Date());
    var j;
    for (j=0; j<100000000; j++) {
        var x;
        x = x + 1;
    }
    console.log(new Date());
    console.log('Function a() Ends');
}
a()
function b() {
console.log('Function B() starts');
   console.log(new Date());
    var a;
    var j;
    for (j=0; j<100000000; j++) {
      a = a + 1;
    }
    console.log(new Date());
    console.log('Function B() Ends');
}
b()

结果显示在我的案例中

1
2
3
4
5
6
7
8
Function a() starts
VM121:3 Thu Apr 12 2018 15:20:26 GMT+0530 (India Standard Time)
VM121:9 Thu Apr 12 2018 15:20:26 GMT+0530 (India Standard Time)
VM121:10 Function a() Ends
VM121:14 Function B() starts
VM121:15 Thu Apr 12 2018 15:20:26 GMT+0530 (India Standard Time)
VM121:21 Thu Apr 12 2018 15:20:26 GMT+0530 (India Standard Time)
VM121:22 Function B() Ends

谢谢 -
MyFavs.in


那么,这取决于你想要实现的目标......如果value假设只是循环块内的临时变量,那么使用第二种形式就更清楚了。它也更合乎逻辑,更冗长。


这里的问题基本上是在循环内声明一个var。试想如果你这样做会发生什么:

1
2
3
var a = 30;
var a = 50;
var a = 60;

你认为这是对的吗?不...因为您不想多次声明变量。当你在循环中声明一个变量时,它不会声明循环运行多少次?当你处于'严格'模式时,它显然会打你。人们在不考虑原始问题的情况下不同意克罗克福德。

因此,最好将变量声明在上面 - 1.为了便于阅读,2。养成良好的习惯。


关于在Linux操作系统上运行Chrome,Firefox和jsperf测试后的性能,在循环中和循环中声明变量之间似乎存在性能差异。这是一个很小的差异,但迭代的数量和变量声明的数量也会加剧这一点。

因此,为了获得最佳性能,我必须建议在循环外声明变量。或者更好地声明你的变量。见例子。

1
2
3
4
5
6
7
8
9
10
11
// inline
for (var ai = 0, al = 100000000, av; ai < al; ai++) {
    av = av + 1;
}

// outside
var bv;
var bl = 100000000;
for (var bi = 0; bi < bl; bi++) {
    bv = bv + 1;
}

注意变量'al'和'av'是如何在for循环声明行中的。这个内联声明为我提供了始终如一的更好的表现。甚至在循环之外的变量声明。性能差异再次非常小。

https://jsperf.com/outside-inline-for-loop-ase/1