エラーが発生しやすい JavaScript に関する一般的な知識ポイントをまとめたもの

阿神
リリース: 2016-12-10 09:22:20
オリジナル
943 人が閲覧しました

1. 変数スコープ

var a = 1; 
function test() { 
    var a = 2; 
    console.log(a); // 2 
} 
test();
ログイン後にコピー

A は上記の関数スコープで宣言および代入されており、コンソールの上にあるため、出力 a は近接原理に従って 2 に等しくなります。

var a = 1;  
function test2() { 
    console.log(a); // undefined 
    var a = 2; 
} 
test2();
ログイン後にコピー

上記の関数スコープ内でaが宣言されて代入されていますが、コンソール配下にあり、a変数は宣言されているものの出力時に値が代入されていないため「未定義」と出力されます。

var a = 1; 
function test3() { 
    console.log(a); // 1 
    a = 2;  
} 
test3();
ログイン後にコピー

上記の関数スコープの A は再宣言ではなく再代入され、コンソール配下にあるため、グローバルスコープの a が出力されます。

let b = 1; 
function test4() { 
    console.log(b); // b is not defined 
    let b = 2; 
} 
test4();
ログイン後にコピー

ES6 let は上記の関数スコープで変数 b を再宣言するために使用されますが、var とは異なり、let には変数を昇格する機能がないため、「b が定義されていません」という出力エラーが報告されます。

function test5() { 
    let a = 1; 
    { 
        let a = 2; 
    } 
    console.log(a); // 1 
} 
test5();
ログイン後にコピー

上記の関数スコープは let を使用して a を 1 として宣言し、ブロックレベルのスコープで a を 2 として宣言しています。コンソールは関数内のブロックレベルのスコープにないため、1 が出力されます。

2. 型の比較

var arr = [], 
    arr2 = [1]; 
console.log(arr === arr2); // false
ログイン後にコピー

上記の 2 つの異なる配列を比較します。コンソールは false です。

var arr = [], 
    arr2 = []; 
console.log(arr === arr2); // false
ログイン後にコピー

上記の 2 つの同一の配列を比較します。2 つの別々の配列は決して等しくないため、コンソールは false になります。

var arr = [], 
    arr2 = {}; 
console.log(typeof(arr) === typeof(arr2)); // true
ログイン後にコピー

上記は typeof を使用して配列とオブジェクトを比較しています。typeof は NULL、配列、オブジェクトを取得するため、型はすべてオブジェクトであるため、コンソールは true になります。

var arr = []; 
console.log(arr instanceof Object); // true 
console.log(arr instanceof Array); // true
ログイン後にコピー

上記では、instanceof を使用して、変数がオブジェクトのインスタンスに属しているかどうかを判断しています。配列も JavaScript のオブジェクトの一種であるため、どちらのコンソールも当てはまります。

3.thisは

var obj = { 
    name: 'xiaoming', 
    getName: function () { 
        return this.name 
    } 
}; 
console.log(obj.getName());  // 'xiaoming'
ログイン後にコピー

を指します。上記のオブジェクトメソッドのThisはオブジェクト自体を指しているので、「xiaoming」が出力されます。

var obj = { 
    myName: 'xiaoming', 
    getName: function () { 
        return this.myName 
    } 
}; 
var nameFn = obj.getName; 
console.log(nameFn()); // undefined
ログイン後にコピー

オブジェクト内のメソッドは上記の変数に割り当てられます。この時点で、メソッド内の this は obj オブジェクトを指すのではなく、window オブジェクトを指すことになるため、コンソールは「未定義」になります。

var obj = { 
    myName: 'xiaoming', 
    getName: function () { 
        return this.myName 
    } 
};  
var obj2 = { 
    myName: 'xiaohua'  
}; 
var nameFn = obj.getName; 
console.log(nameFn.apply(obj2)); // 'xiaohua'
ログイン後にコピー

obj オブジェクトのメソッドは上記の変数 nameFn にも代入されていますが、これは apply メソッドを通じて obj2 オブジェクトを指しているため、最終的なコンソールは 'xiaohua' になります。

4. 関数パラメータ

function test6() { 
    console.log(arguments); // [1, 2] 
} 
test6(1, 2);
ログイン後にコピー

上記は関数内の引数オブ​​ジェクトを使用して、関数に渡されるパラメータ配列を取得するため、出力配列は [1, 2] になります。

function test7 () { 
    return function () { 
        console.log(arguments); // 未执行到此,无输出 
    } 
} 
test7(1, 2);
ログイン後にコピー

上記でもパラメータを取得するために引数を使用していますが、test7(1, 2) は戻り値として関数を実行しないため、test7(1, 2)(3, 4) が実行された場合、出力はありません。 ,4]が出力されます。

var args = [1, 2]; 
function test9() { 
    console.log(arguments); // [1, 2, 3, 4] 
} 
Array.prototype.push.call(args, 3, 4); 
test9(...args);
ログイン後にコピー

上記は Array.prototype.push.call() メソッドを使用して 3 と 4 を args 配列に挿入し、ES6 拡張演算子 (...) を使用して配列を展開して test9 に渡します。コンソールは[1、2、3、4]です。

5. クロージャの問題

var elem = document.getElementsByTagName('div'); // 如果页面上有5个div 
for(var i = 0; i < elem.length; i++) { 
    elem[i].onclick = function () { 
        alert(i); // 总是5 
    }; 
}
ログイン後にコピー

上記は、クリック イベントをトリガーしたときに i の値がすでに 5 であるため、div をクリックしたときに表示される値は常に 5 になる、非常に一般的なクロージャの問題です。次のメソッドを使用できます。 解決策:

var elem = document.getElementsByTagName(&#39;div&#39;); // 如果页面上有5个div 
for(var i = 0; i < elem.length; i++) { 
    (function (w) { 
        elem[w].onclick = function () { 
            alert(w); // 依次为0,1,2,3,4 
        }; 
    })(i); 
}
ログイン後にコピー

バインドされたクリック イベントの外側で即時実行関数をカプセル化し、関数に i を渡します。

6. オブジェクトのコピーと代入

var obj = { 
    name: &#39;xiaoming&#39;, 
    age: 23 
}; 
 
var newObj = obj; 
newObj.name = &#39;xiaohua&#39;; 
console.log(obj.name); // &#39;xiaohua&#39; 
console.log(newObj.name); // &#39;xiaohua&#39;
ログイン後にコピー

上記では、obj オブジェクトを newObj オブジェクトに代入して、newObj の name 属性を変更しましたが、これは、obj オブジェクトの name 属性も改ざんされているためです。実際に取得されたオブジェクト これは単なるメモリアドレスであり、実際のコピーではないため、obj オブジェクトは改ざんされます。

var obj2 = { 
    name: &#39;xiaoming&#39;, 
    age: 23 
}; 
var newObj2 = Object.assign({}, obj2, {color: &#39;blue&#39;}); 
newObj2.name = &#39;xiaohua&#39;; 
console.log(obj2.name); // &#39;xiaoming&#39; 
console.log(newObj2.name); // &#39;xiaohua&#39; 
console.log(newObj2.color); // &#39;blue&#39;
ログイン後にコピー

上記の Object.assign() メソッドを使用してオブジェクトのディープ コピーを実行すると、ソース オブジェクトが改ざんされる可能性を回避できます。 Object.assign() メソッドは、ソース オブジェクト自体の列挙可能なプロパティを任意の数でターゲット オブジェクトにコピーし、ターゲット オブジェクトを返すことができるためです。

var obj3 = { 
    name: &#39;xiaoming&#39;, 
    age: 23 
}; 
var newObj3 = Object.create(obj3); 
newObj3.name = &#39;xiaohua&#39;; 
console.log(obj3.name); // &#39;xiaoming&#39; 
console.log(newObj3.name); // &#39;xiaohua&#39;
ログイン後にコピー

Object.create() メソッドを使用してオブジェクトをコピーすることもできます。Object.create() メソッドは、指定されたプロトタイプ オブジェクトとプロパティを使用して新しいオブジェクトを作成できます。

結論

JavaScriptの学習は長いプロセスであり、一夜にして達成できるものではありません。この記事で紹介したポイントが、JavaScript を学習している学生の皆さんが JavaScript 構文をより深く理解して習得し、寄り道を避けるのに役立つことを願っています。


関連ラベル:
ソース:php.cn
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
最新の問題
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート