ホームページ >ウェブフロントエンド >jsチュートリアル >JavaScriptによるシャローコピーとディープコピーの詳細説明

JavaScriptによるシャローコピーとディープコピーの詳細説明

高洛峰
高洛峰オリジナル
2017-01-03 15:52:461384ブラウズ

以下のエディターでは、JavaScript のオブジェクト指向のディープ コピーとシャロー コピーについて簡単に説明します。編集者はこれがとても良いと思ったので、参考として共有します。

1. 浅いコピー: 参照をコピーします。すべての参照オブジェクトはデータのコピーを指し、このデータを変更できます。

2. ディープコピー (複合): 非基本型変数の場合は、基本型変数まで再帰的にコピーします。

理解を深めるための簡単な図を以下に示します:

JavaScriptによるシャローコピーとディープコピーの詳細説明

1. 配列の深いコピーと浅いコピー

JavaScript を使用して配列を操作する場合、多くの場合、配列をバックアップする必要があることが事実で証明されています。単純な場合、それを他の変数に代入すると、そのうちの 1 つを変更するだけで済み、他の変数も変更されてしまい、問題が発生します。

var arr = ["One","Two","Three"];
var arrto = arr;
arrto[1] = "test";
document.writeln("数组的原始值:" + arr + "<br />");//Export:数组的原始值:One,test,Three
document.writeln("数组的新值:" + arrto + "<br />");//Export:数组的新值:One,test,Three

上記のような直接代入メソッドは、多くの場合、私たちが望む結果ではありません。arr の値が変更されないことです。

方法 1: js のスライス関数

var arr = ["One","Two","Three"];
var arrtoo = arr.slice(0);
arrtoo[1] = "set Map";
document.writeln("数组的原始值:" + arr + "<br />");//Export:数组的原始值:One,Two,Three
document.writeln("数组的新值:" + arrtoo + "<br />");//Export:数组的新值:One,set Map,Three

方法 2: js の concat メソッド

var arr = ["One","Two","Three"];
var arrtooo = arr.concat();
arrtooo[1] = "set Map To";
document.writeln("数组的原始值:" + arr + "<br />");//Export:数组的原始值:One,Two,Three
document.writeln("数组的新值:" + arrtooo + "<br />");//Export:数组的新值:One,set Map To,Three

2 番目に、オブジェクトの深いコピーと浅いコピー

var a={name:&#39;yy&#39;,age:26};
var b=new Object();
b.name=a.name;
b.age=a.age;
a.name=&#39;xx&#39;;
console.log(b);//Object { name="yy", age=26}
console.log(a);//Object { name="xx", age=26}

は、オブジェクトのプロパティを走査し、それを新しいオブジェクトに割り当てます。

var deepCopy= function(source) { 
var result={};
for (var key in source) {
 result[key] = typeof source[key]===&#39;object&#39;? deepCoyp(source[key]): source[key];
 } 
 return result; 
}

jQuery の例を示します:

jQuery.extend = jQuery.fn.extend = function() {//1.将extend方法扩展到JQ(函数)下边:扩展静态方法
 //2. jQuery.fn.extend 把extend扩展到jq.fn下 且jQuery.fn = jQuery.prototype 扩展实例方法
 // 1.2.功能相似
 var options, name, src, copy, copyIsArray, clone, //定义一些变量
 target = arguments[0] || {}, 
 //目标元素是【0】第一个元素$.extend( a , { name : &#39;hello&#39; } , { age : 30 } );
 i = 1, //第一个元素的位置
 length = arguments.length,//第一个个对象的元素
 deep = false; //是否是深拷贝 默认 false不是
 
 // Handle a deep copy situation 看是不是深拷贝情况
 if ( typeof target === "boolean" ) { //是布尔值 
 deep = target;
 target = arguments[1] || {}; //目标元素是第二个$.extend( true , a , b )
 // skip the boolean and the target
 i = 2;
 }
 
 // Handle case when target is a string or something (possible in deep copy) 看参数正确不
 if ( typeof target !== "object" && !jQuery.isFunction(target) ) {
 // 当目标不是对象或者不是函数的时候
 target = {}; //变成一个空的jason
 }
 
 // extend jQuery itself if only one argument is passed看是不是插件情况
 if ( length === i ) { //只写了一个对象 要把这个对象扩展到jq源码上 静态方法 或者是实例方法
 target = this; //this 是$ 或者 $();
 --i;
 }
// 可能有多个对象情况
 for ( ; i < length; i++ ) {
 // Only deal with non-null/undefined values
 if ( (options = arguments[ i ]) != null ) {//看后边的对象是否都有值
 // Extend the base object
 for ( name in options ) {
 src = target[ name ];
 copy = options[ name ];
 
 // Prevent never-ending loop
 if ( target === copy ) {//防止循环引用 
  continue;//跳出本次循环继续执行
  // $.extend( a , { name : a } ) );循环引用 a也是一个对象
 }
 
 // Recurse if we&#39;re merging plain objects or arrays深拷贝
 if ( deep && copy && ( jQuery.isPlainObject(copy) || (copyIsArray = jQuery.isArray(copy)) ) ) {
  // 是深拷贝 且需有var b = { name : { age : 30 } }; 且b必须是对象自变量(jason) 或者是个数组
   
  //递归
  if ( copyIsArray ) { //数组
  copyIsArray = false;
  clone = src && jQuery.isArray(src) ? src : []; //定义一个空数组
 
  } else {//jason
  clone = src && jQuery.isPlainObject(src) ? src : {};//看原有的属性有没有且是不是jason定义一个空jason
  }
  // var a = { name : { job : &#39;it&#39; } }; 看有没有原有的属性 有的话在原有的上边添加
  // var b = { name : {age : 30} };
  // $.extend( true , a , b );//a继承b
  // console.log( a ); a{ name:{ job : &#39;it&#39; ,age : 30}} 如果只有一个{} 则只有,age : 30
  // Never move original objects, clone(a) them
  target[ name ] = jQuery.extend( deep, clone, copy ); 
  //调用函数本身进行进一步的递归处理 
 
 // Don&#39;t bring in undefined values浅拷贝
 } else if ( copy !== undefined ) {
  target[ name ] = copy; //直接复制因为里边没有对象
 }
 }
 }
 }
 
 // Return the modified object
 return target;
};

以上がこの記事の全内容です。皆様の学習に役立つことを願っております。また、皆様にも PHP 中国語 Web サイトをサポートしていただければ幸いです。

JavaScript の浅いコピーと深いコピーの詳細な説明については、PHP 中国語 Web サイトに注目してください。

声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。