間違いやすい JavaScript の知識ポイントをまとめました。

怪我咯
リリース: 2017-04-05 14:23:26
オリジナル
1570 人が閲覧しました

前書き

この記事は、私が JavaScript を学習する過程で収集し、整理した間違いやすい知識ポイントについて説明したもので、変数のスコープ、型の比較、このポインター、関数のパラメーター、クロージャの問題の 6 つの側面について説明します。 、オブジェクトのコピーと割り当てについて、浅いところから深いところまで紹介して説明しますが、これには ES6 の知識ポイントも含まれます。

JavaScriptの知識ポイント

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 つの同一の配列を比較してください。配列と配列の間の比較は常に false であるため、コンソールは 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('p'); // 如果页面上有5个p

for(var i = 0; i < elem.length; i++) {
    elem[i].onclick = function () {
        alert(i); // 总是5
    };
}
ログイン後にコピー

上記は、クリック イベントをトリガーしたときに i の値がすでに 5 であるため、任意の p をクリックしたときに表示される値は常に 5 です。次のメソッドを使用できます。 解決策:

var elem = document.getElementsByTagName(&#39;p&#39;); // 如果页面上有5个p

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 構文をより深く理解して習得し、寄り道を避けるのに役立つことを願っています。

以上が間違いやすい JavaScript の知識ポイントをまとめました。の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

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