開發中,打斷對象間的引用關系,只想下個副本的情況無處不在,clone一個對象就在所難免了。
JavaScript中,簡單的方法就是用JSON函數,將對象stringify成字符串,再parse成一個新對象。要麼就是從網上搜個代碼,開源社區裡面clone的代碼還是有不少的。
代碼雖然可以找得到,但,東西永遠是別人的,動手學著碼永遠是個不變的主題。
自己寫了兩個克隆的函數:
cloneOwn:克隆自定義對象的自有屬性,不包括繼承的屬性,屬性可以是基本數據類型和數組,自定義的對象,可以制定要克隆的屬性名稱列表。
cloneArray: 克隆數組,數組內的元素可以是對象,基本類型。
代碼如下:
//第一個參數是被克隆的對象,第二個參數是需要克隆的屬性列表
function cloneOwn() {
var obj = arguments[0];
if (typeof obj === 'undefined' || obj === null)
return {};
if (typeof obj !== 'object')
return obj;
//第二個參數是屬性名稱列表,就采用該列表進行刷選
//否則就克隆所有屬性
var attrs = arguments[1];
var enable_spec_attr = true;
if (!(attrs instanceof Array)) {
//console.log(attrs);
attrs = obj;
enable_spec_attr = false;
}
var result = {};
var i;
for (i in attrs) {
attr = enable_spec_attr? attrs[i]: i;
//console.log(attr);
if (obj.hasOwnProperty(attr)) {
if (obj[attr] instanceof Array) {
result[attr] = cloneArray(obj[attr]);
}
else if (typeof obj[attr] === 'object') {
result[attr] = cloneOwn(obj[attr]);
} else {
result[attr] = obj[attr];
}
}
}
return result;
}
代碼如下:
//克隆數組
function cloneArray(array) {
if (typeof array === 'undefined' || array === null)
return [];
if (!(array instanceof Array))
return [];
result = [];
var i;
for(i in array) {
if (typeof array[i] !== 'object') {
result[i] = array[i];
continue;
}
//clone object
result[i] = cloneOwn(array[i]);
}
return result;
}
調用
1.常規克隆自定義對象:
代碼如下:
var a = {
name:'frank',
age:20
};
var b= cloneOwn(a);
2.指定克隆的屬性
代碼如下:
var a = {
name:'frank',
age:20,
address:'any where'
};
var b = cloneOwne(a, ['name', 'age']);
3.克隆內含有數組屬性的自定義對象
代碼如下:
var a = {
name: 'kxh',
age: 20,
books: ['hai','ho','ali'],
likes: [
{wname: 'kaili', wage: 81, fav: "aaaaa"},
{wname: 'seli', wage: 82, fav: "bbb"},
{wname: 'ailun', wage: 83, fav: "ccc"},]
};
var b = cloneOwne(a);
4.克隆數組,內含有自定義對象
代碼如下:
var a = [
{
name:'frank',
age:20
},
{
name:'leon',
age:30
}
];
var b = cloneArray(a);
上面的代碼還是有很多問題的,比如,內置對象的克隆就存在點問題,例如datatime類型。
問題管問題,這樣一個學習過程也是要有的。