JavaScript のオブジェクト コピー メソッドの詳細な分析 (コード付き)

奋力向前
リリース: 2021-08-20 09:56:39
転載
1936 人が閲覧しました

前回の記事「JavaScript「jquery継承」の使い方で知っておきたいこと(詳細コード解説)」では、JavaScript「jquery継承」の使い方について学びました。以下の記事ではJSでオブジェクトをコピーする方法を紹介していますので、困っている方は参考にしてみてください。

JavaScript のオブジェクト コピー メソッドの詳細な分析 (コード付き)

javascript でオブジェクトをコピーする場合、最初に思い浮かぶのは Object.assign()## です。

#JSON.parse(JSON.stringify())、および ES6 の展開演算子 [...]

js#=では、operator はオブジェクトのコピーを作成できないため、これはオブジェクトへの参照にすぎませんOperator

var x = {
  a: 1,
  b: 2,
};
y = x;
x.a = 10;
console.log(x); //{a:10, b:2}
console.log(y); //{a:10, b:2}
ログイン後にコピー

したがって、オブジェクト操作を実行する場合、等号演算子 (

=

) は推奨されません

Object.assign()

var x = {
  a: 1,
  b: 2,
};
y = Object.assign({}, x);
x.a = 10;
console.log(x); //{a:10, b:2}
console.log(y); //{a:1, b:2}
ログイン後にコピー
一見すると、次のようになります。ではありません 欲しいものは欲しい結果なので例外が見つかります オブジェクトの構造をもう少し複雑にして見てみると

var x = {
  a: 1,
  b: 2,
  c: {
    d: 3,
  },
};
y = Object.assign({}, x);

x.a = 5;
console.log(x); //{a:5, b:2, c:{d:3}}
console.log(y); //{a:5, b:2, c:{d:3}}

x.c.d = 10;
console.log(x); //{a:5, b:2, c:{d:10}}
console.log(y); //{a:5, b:2, c:{d:10}}
ログイン後にコピー

ですが、この時に落とし穴が見つかります。それは証明されています

Object.assign()

オブジェクトの浅いコピーのみを実装します

Object.assign()

もう 1 つ注意すべきことは、列挙不可能であることです。プロトタイプ チェーンに属性を持つオブジェクトはコピーできません。コードを見てください。 : <div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:php;toolbar:false">var x = { a: 1, }; var y = Object.create(x, { b: { value: 2, }, c: { value: 3, enumerable: true, }, }); var z = Object.assign({}, y); console.log(z); //{c:3}</pre><div class="contentsignin">ログイン後にコピー</div></div>

z

の値を取得するのは驚くべきことです。なぜなら xy のプロトタイプ チェーンであるため、#xx はコピーされません属性 b

は列挙不可能なプロパティであるため、コピーされません

c

のみが列挙可能な説明を持ち、列挙できるため、コピーできます

上記の落とし穴も非常にうまく解決でき、以下のようになります。深いコピーJSON.parse(JSON.stringify())

浅いコピーの落とし穴への解決策

var x = {
  a: 1,
  b: 2,
  c: {
    d: 3,
  },
};
y = JSON.parse(JSON.stringify(x));
x.a = 5;
x.c.d = 10;
console.log(x); //{a:5, b:2, c:{d:10}}
console.log(y); //{a:1, b:2, c:{d:3}}
ログイン後にコピー
もちろん、通常のオブジェクトの場合、このコピー方法は基本的には完璧なので、落とし穴はどこにありますか
var x = {
  a: 1,
  b: function b() {
    return "2";
  },
};
y = JSON.parse(JSON.stringify(x));
z = Object.assign({}, x);

console.log(y); //{a:1}
console.log(z); //{a:1, b:function b(){return &#39;2&#39;}}
ログイン後にコピー

結果から、

Object.assign()

はメソッド

JSON.parse(JSON.stringify())

をコピーできます。 cannot

2 つ目の落とし穴を見てみましょう。落とし穴:

var x = {
  a: 1,
  b: {
    c: 2,
    d: 3,
  },
};

x.c = x.b;
x.d = x.a;
x.b.c = x.c;
x.b.d = x.d;

var y = JSON.parse(JSON.stringify(x));
console.log(x);

/*
Uncaught TypeError: Converting circular structure to JSON

    at JSON.stringify (<anonymous>)
    at <anonymous>:8:25

*/
ログイン後にコピー
がエラーを報告し、結果は JSON.parse(JSON.stringify()),

cannot を示しています。循環参照オブジェクトをコピーします

見てみましょう

Object.assign()

var x = {
  a: 1,
  b: {
    c: 2,
    d: 3,
  },
};

x.c = x.b;
x.d = x.a;
x.b.c = x.c;
x.b.d = x.d;

var y = Object.assign({}, x);
console.log(x);
/*
[object Object]{
a:1, 
b:[object, Object], 
d:[object, Object], 
d:1
}
*/
ログイン後にコピー

展開演算子を使用します [

... ]

オブジェクト リテラルの展開演算子は現在

ECMAScript のフェーズ 3 提案により、オブジェクトのコピーがより簡単になります
var x = [
  "a",
  "b",
  "c",
  "d",
  {
    e: 1,
  },
];
var y = [...x];
console.log(y); //[&#39;a&#39;, &#39;b&#39;, &#39;c&#39;, &#39;d&#39;, {&#39;e&#39;:1}]

var m = {
  a: 1,
  b: 2,
  c: ["d", "e"],
};
var n = {
  ...m,
};
console.log(n); //{a:1, b:2, c:[&#39;d&#39;, &#39;e&#39;]}
ログイン後にコピー

スプレッド演算子も浅いコピーであることに注意してください。では、オブジェクトをコピーすることは本当に難しいのでしょうか?

自分の車輪を発明する:

function copy(x) {
  var y = {};
  for (m in x) {
    y[m] = x[m];
  }
  return y;
}
var o = {
  a: 1,
  b: 2,
  c: {
    d: 3,
    e: 4,
  },
};
var p = copy(o);
ログイン後にコピー

これを実行しても大きな問題はないはずだと言う人もいます。

var x = {};

Object.defineProperty(x, "m", {
  value: 5,
  writable: false,
});

console.log(x.m); //5
x.m = 25; //这一步没报错,但是也没执行
console.log(x.m); //5
ログイン後にコピー
このように、展開演算子はここでオブジェクトをコピーするときに落とし穴に遭遇することになります。

落とし穴はどこにでもあり、それらを防ぐのは困難です。これを書いている時点で、完全にはリストされていない落とし穴がまだたくさんあると思います。

I'詳細は後ほど書きます

##[終了]

推奨学習:

JavaScript ビデオ チュートリアル

以上がJavaScript のオブジェクト コピー メソッドの詳細な分析 (コード付き)の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

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