由jQuery深拷贝引发的学习

先前对javascript的继承学的很模糊,由此我就百度各种文章,然文章千奇百怪,虽不乏精妙之言,却独无对吾之口味,由此从jquery
中的extend方法开始学起,首先上源码copy自jQuery1.7版本

jQuery.extend = jQuery.fn.extend = function() {
var options, name, src, copy, copyIsArray, clone,
target = arguments[0] || {},
i = 1,
length = arguments.length,
deep = false;

// Handle a deep copy situation
if ( typeof target === "boolean" ) {
deep = target;
target = arguments[1] || {};
// 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 = {};
}

// extend jQuery itself if only one argument is passed
if ( length === i ) {
target = 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;
}

// Recurse if we're merging plain objects or arrays
if ( deep && copy && ( jQuery.isPlainObject(copy) || (copyIsArray = jQuery.isArray(copy)) ) ) {
if ( copyIsArray ) {
copyIsArray = false;
clone = src && jQuery.isArray(src) ? src : [];

} else {
clone = src && jQuery.isPlainObject(src) ? src : {};
}

// Never move original objects, clone them
target[ name ] = jQuery.extend( deep, clone, copy );

// Don't bring in undefined values
} else if ( copy !== undefined ) {
target[ name ] = copy;
}
}
}
}

// Return the modified object
return target;
};


笼统看一遍,很难产生什么直观的感觉,先看该方法的定义
合并两个或更多的对象汇集成到第一个对象中

jQuery.extend([deep,]Target[,object1,][objectN]);

 
 可以去看一下该函数的说明,有以下几种用法

jQuery.extend(object);

jQuery.extend(object1,object2,objectN);

jQuery.extend(bool,object1);

jQuery.extend(bool,object1,object2,objectN);

 
我们从上面的方法来分析jQuery是怎么处理的,首先是jQuery.extend(object)。一个参数的时候,目标对象将会被jquery本身代替

//对象1
var obj1={name:"hello",sex:"man",fruits:["1","2"],bb:{name:"gg"}};

$.extend(obj1);
document.write($.name);

//result:
//hello

对象的属性已经被附加到jQuery中了,看看其实现

//关键之处,参数的长度如果和预设的长度相等的话,那么目标对象将会被改变为自身
if ( length === i ) {
target = this;
--i;
}

然后就是对象的拷贝

//这里的拷贝只是简单的拷贝,通常说的浅拷贝,与深拷贝的不同继续看下面
else if ( copy !== undefined ) {
target[ name ] = copy;
}

那么深拷贝与浅拷贝有什么不同呢?看例子

浅拷贝

//对象1
var obj1={name:"hello",sex:"man",fruits:["1","2"],bb:{name:"gg"}};

$.extend(obj1);
obj1.name="hello is changed!";
obj1.fruits[0]="1 is changed";
obj1.bb.name="gg is changed";
document.write($.name+"<br>"+$.fruits[0]+"<br>"+$.bb.name);

//results:
//hello
//1 is changed
//gg is changed
//结果表明obj1的array与object类型的数据被改变后会影响到目标对象,而值类型
//确不会,现在是不是有点意思了,深拷贝就是要排除这个影响,让他们互不侵犯

深拷贝

//对象1
var obj1={name:"hello",sex:"man",fruits:["1","2"],bb:{name:"gg"}};

$.extend(true,obj1); //调用时候变成了2个参数,第一个布尔值就是深拷贝标识
obj1.name="I changed!";
obj1.fruits[0]="1 is changed";
obj1.bb.name="gg is changed";
document.write($.name+"<br>"+$.fruits[0]+"<br>"+$.bb.name);

//results:
//hello
//1
//gg
//现在源对象的改变再也不会影响到目标对象了,下面看看jquery是怎么实现的

关键点把对象或数组的拷贝转化为值的拷贝

 if ( copyIsArray ) {
copyIsArray = false;
clone = src && jQuery.isArray(src) ? src : [];
//对象是否是数组,是的话记录对象本身,否则创建一个新的空数组
} else {
clone = src && jQuery.isPlainObject(src) ? src : {};
//对象是否为object,是的话记录对象本身,否则创建一个新的空对象
}

//关键之处,对array或object进行迭代,把他们的值进行拷贝
target[ name ] = jQuery.extend( deep, clone, copy );

现在再看jQuery.extend(object1,object2,objectN);、jQuery.extend(bool,object1,object2,objectN);这两种用法就很容易理解是怎么实现的了。

//对象1
var obj1={name:"hello",sex:"man",fruits:["1","2"],bb:{name:"gg"}};
//对象2
var obj2={name:"world",sex:"woman",fruits:["3","4"],bb:{name:"mm"}};
//对象3
var obj3={name:"!",sex:"undefind",fruits:["5","6"],bb:{name:"gm"}};

$.extend(obj1,obj2,obj3);
obj3.bb.name="gm is changed"; //如果是obj2.bb.name="aa";是不会影响obj1的,因为参数是依次遍历的哦,前面的会被后面的覆盖
for(var i in obj1){
if(jQuery.type(obj1[i])=="array"||jQuery.type(obj1[i])=="object")
{
for(var i2 in obj1[i]){document.write("<br>"+obj1[i][i2]);}
}else{
document.write("<br>"+obj1[i]);

}
}
//results:
//!
//undefind
//5
//6
//gm is changed 这里是浅拷贝的缘故哦

$.extend(true,obj1,obj2,obj3);   深拷贝
//results:
//!
//undefined
//5
//6
//gm

以上就是jquery用来实现“继承”的方法,和传统的OO继承比较,不同的是,“子类”的属性、方法会被“父类”的属性、方法给覆盖,这让我有点费解,也许

是被先入为主的思想所左右了,也奇怪jquery的开发人员怎么不搞的传统点,也许他们觉得这样也就够了,也许我没理解透彻。

扯着扯着,又得写下,传统的OO继承在JS中的实现

看个例子

//将Object.prototype上的toString方法重写,以便测试
Object.prototype.toString=function(){document.write("I am object");};

//父类
function ParentFunction(){
this.toString=function (){document.write("I am parent");}
}
ParentFunction.prototype.toString=function(){document.write("I am parent prototype");};

//子类
function SubFunction(){
this.toString=function(){document.write("I am sub");}
}


//SubFunction继承了ParentFunction,通过将原型new一个ParentFunction对象来建立关系
SubFunction.prototype=new ParentFunction();
//由于上面的prototype被重写了,而我们又必须让原型链完整,构造函数属性指向SubFunction
SubFunction.prototype.constructor=SubFunction;

//可以再在prototype上添加方法
//
SubFunction.prototype.toString=function(){document.write("I am sub prototype");};

var test=new SubFunction();
test.toString(); //result: I am sub

结果貌似不怎么明显表达意思,那么把SubFunction里的方法去掉

//result: I am parent

有点意思了,方法调用的是父类的本地代码里的方法,再将ParentFunction里的方法去掉

//result: I am parent prototype

调用的是父类prototype上的方法,继续把ParentFunction.prototype.toString去掉如何?

//result: I am object 

调用了Object对象上的方法,别奇怪,那是因为Object是所有对象的父类。

上个图,也许能理解的清楚点


学习做此小记,最后欢快一下。

“辛苦十几年,你也不过是红警里500块一个的工程师,一条狗咬死一片的那种。。。”

posted @ 2012-03-16 15:50  一文钱  阅读(3674)  评论(7编辑  收藏  举报