この記事の情報は、古いスクリプトを理解するのに役立ちます。
それは、新しいコードの書き方ではありません。
変数に関する最初の章で、変数を宣言する3つの方法について述べました。
let
const
var
var
宣言はlet
と似ています。ほとんどの場合、let
をvar
に置き換えたり、その逆も同様に置き換えて、動作することを期待できます。
var message = "Hi";
alert(message); // Hi
しかし、内部的にはvar
は非常に古い時代に由来する、非常に異質な存在です。一般的には現代のスクリプトでは使用されませんが、古いスクリプトにはまだ潜んでいます。
そのようなスクリプトに遭遇する予定がない場合は、この章をスキップしたり、後回しにしたりしても構いません。
一方で、古いスクリプトをvar
からlet
に移行する場合、奇妙なエラーを避けるために違いを理解することが重要です。
「var」にはブロックスコープがない
var
で宣言された変数は、関数スコープまたはグローバルスコープのいずれかになります。それらはブロックを通して可視です。
例えば
if (true) {
var test = true; // use "var" instead of "let"
}
alert(test); // true, the variable lives after if
var
はコードブロックを無視するため、グローバル変数test
が得られます。
var test
の代わりにlet test
を使用した場合は、変数はif
の中でしか可視になりません。
if (true) {
let test = true; // use "let"
}
alert(test); // ReferenceError: test is not defined
ループも同様です。var
はブロックまたはループローカルにすることはできません。
for (var i = 0; i < 10; i++) {
var one = 1;
// ...
}
alert(i); // 10, "i" is visible after loop, it's a global variable
alert(one); // 1, "one" is visible after loop, it's a global variable
コードブロックが関数内にある場合、var
は関数レベルの変数になります。
function sayHi() {
if (true) {
var phrase = "Hello";
}
alert(phrase); // works
}
sayHi();
alert(phrase); // ReferenceError: phrase is not defined
見てわかるように、var
はif
、for
、または他のコードブロックを貫通します。それは、JavaScriptのずっと昔には、ブロックにレキシカル環境がなく、var
はその残存物であるためです。
「var」は再宣言を許容する
同じスコープ内でlet
を使って同じ変数を2回宣言すると、エラーになります。
let user;
let user; // SyntaxError: 'user' has already been declared
var
では、変数を何度でも再宣言できます。すでに宣言されている変数でvar
を使用すると、それは無視されます。
var user = "Pete";
var user = "John"; // this "var" does nothing (already declared)
// ...it doesn't trigger an error
alert(user); // John
「var」変数は使用するよりも下で宣言できる
var
宣言は、関数の開始時(グローバルの場合はスクリプトの開始時)に処理されます。
言い換えれば、var
変数は、定義がどこにあるかに関係なく(定義がネストされた関数にない場合)、関数の最初から定義されます。
したがって、このコードは
function sayHi() {
phrase = "Hello";
alert(phrase);
var phrase;
}
sayHi();
…技術的にはこれと同じです(var phrase
を上に移動)
function sayHi() {
var phrase;
phrase = "Hello";
alert(phrase);
}
sayHi();
…またはこれと同じです(コードブロックは無視されることを覚えておいてください)
function sayHi() {
phrase = "Hello"; // (*)
if (false) {
var phrase;
}
alert(phrase);
}
sayHi();
人々は、この動作を「ホイスティング」(持ち上げ)と呼ぶこともあります。なぜなら、すべてのvar
は関数の上部に「ホイスト」(持ち上げ)されるからです。
したがって、上記の例では、if (false)
ブランチは決して実行されませんが、それは問題ではありません。その中のvar
は関数の最初に処理されるため、(*)
の時点では変数は存在します。
宣言はホイストされますが、代入はホイストされません。
それが最もよく示されるのは、例です。
function sayHi() {
alert(phrase);
var phrase = "Hello";
}
sayHi();
var phrase = "Hello"
という行には、2つのアクションがあります。
- 変数宣言
var
- 変数代入
=
。
宣言は関数実行の開始時に処理されます(「ホイスト」)、しかし代入はそれが現れる場所で常に機能します。したがって、コードは基本的にこのように機能します。
function sayHi() {
var phrase; // declaration works at the start...
alert(phrase); // undefined
phrase = "Hello"; // ...assignment - when the execution reaches it.
}
sayHi();
すべてのvar
宣言は関数の開始時に処理されるため、どこからでも参照できます。ただし、変数は代入されるまで未定義です。
上記の両方の例で、変数phrase
が存在するため、alert
はエラーなしで実行されます。ただし、その値はまだ割り当てられていないため、undefined
と表示されます。
IIFE
過去には、var
しかなく、ブロックスコープの可視性がないため、プログラマーはそれをエミュレートする方法を考案しました。彼らが行ったのは、「即時呼び出し関数式」(IIFEと略されます)と呼ばれるものでした。
これは今日使用するものではありませんが、古いスクリプトで見つけることができます。
IIFEは次のようになります。
(function() {
var message = "Hello";
alert(message); // Hello
})();
ここで、関数式が作成され、即座に呼び出されます。したがって、コードはすぐに実行され、独自のプライベート変数を持っています。
関数式は括弧(function {...})
で囲まれています。なぜなら、JavaScriptエンジンがメインコードで"function"
を検出すると、それを関数宣言の開始として理解するためです。しかし、関数宣言には名前が必要なので、この種のコードはエラーになります。
// Tries to declare and immediately call a function
function() { // <-- SyntaxError: Function statements require a function name
var message = "Hello";
alert(message); // Hello
}();
「わかりました、名前を追加しましょう」と言っても、JavaScriptは関数宣言をすぐに呼び出すことを許可していないため、うまくいきません。
// syntax error because of parentheses below
function go() {
}(); // <-- can't call Function Declaration immediately
したがって、関数の周りの括弧は、JavaScriptに関数が別の式のコンテキストで作成されていることを示すためのトリックであり、したがって、それは関数式です。名前は必要なく、すぐに呼び出すことができます。
JavaScriptに、関数式を意味することを伝えるための括弧以外の他の方法も存在します。
// Ways to create IIFE
(function() {
alert("Parentheses around the function");
})();
(function() {
alert("Parentheses around the whole thing");
}());
!function() {
alert("Bitwise NOT operator starts the expression");
}();
+function() {
alert("Unary plus starts the expression");
}();
上記のすべての場合で、関数式を宣言してすぐに実行します。もう一度注意しておきましょう。今日では、このようなコードを書く理由はありません。
まとめ
var
とlet/const
には2つの主な違いがあります。
var
変数にはブロックスコープがなく、その可視性は現在の関数、または関数外で宣言された場合はグローバルにスコープされます。var
宣言は、関数の開始時(グローバルの場合はスクリプトの開始時)に処理されます。
グローバルオブジェクトに関連するもう1つの非常に小さな違いがあり、それは次の章で説明します。
これらの違いにより、var
はほとんどの場合let
よりも劣っています。ブロックスコープの変数は非常に素晴らしいものです。そのため、let
はかなり前に標準に導入され、現在では変数を宣言するための主要な方法(const
と合わせて)になっています。
コメント
<code>
タグを使用します。数行の場合は、<pre>
タグで囲み、10行以上の場合は、サンドボックス(plnkr、jsbin、codepen…)を使用します。