JavaScritp 专题系列第七篇,讲解如何从零实现一个 jQuery 的 extend 函数git
jQuery 的 extend 是 jQuery 中应用很是多的一个函数,今天咱们一边看 jQuery 的 extend 的特性,一边实现一个 extend!github
先来看看 extend 的功能,引用 jQuery 官网:数组
Merge the contents of two or more objects together into the first object.函数
翻译过来就是,合并两个或者更多的对象的内容到第一个对象中。ui
让咱们看看 extend 的用法:spa
jQuery.extend( target [, object1 ] [, objectN ] )复制代码
第一个参数 target,表示要拓展的目标,咱们就称它为目标对象吧。翻译
后面的参数,都传入对象,内容都会复制到目标对象中,咱们就称它们为待复制对象吧。code
举个例子:对象
var obj1 = {
a: 1,
b: { b1: 1, b2: 2 }
};
var obj2 = {
b: { b1: 3, b3: 4 },
c: 3
};
var obj3 = {
d: 4
}
console.log($.extend(obj1, obj2, obj3));
// {
// a: 1,
// b: { b1: 3, b3: 4 },
// c: 3,
// d: 4
// }复制代码
当两个对象出现相同字段的时候,后者会覆盖前者,而不会进行深层次的覆盖。blog
结合着上篇写得 《JavaScript专题之深浅拷贝》,咱们尝试着本身写一个 extend 函数:
// 初版
function extend() {
var name, options, src, copy;
var length = arguments.length;
var i = 1;
var target = arguments[0];
for (; i < length; i++) {
options = arguments[i];
if (options != null) {
for (name in options) {
src = target[name];
copy = options[name];
if (copy !== undefined){
target[name] = copy;
}
}
}
}
return target;
};复制代码
那如何进行深层次的复制呢?jQuery v1.1.4 加入了一个新的用法:
jQuery.extend( [deep], target, object1 [, objectN ] )复制代码
也就是说,函数的第一个参数能够传一个布尔值,若是为 true,咱们就会进行深拷贝,false 依然当作浅拷贝,这个时候,target 就日后移动到第二个参数。
仍是举这个例子:
var obj1 = {
a: 1,
b: { b1: 1, b2: 2 }
};
var obj2 = {
b: { b1: 3, b3: 4 },
c: 3
};
var obj3 = {
d: 4
}
console.log($.extend(true, obj1, obj2, obj3));
// {
// a: 1,
// b: { b1: 3, b2: 2, b3: 4 },
// c: 3,
// d: 4
// }复制代码
由于采用了深拷贝,会遍历到更深的层次进行添加和覆盖。
咱们来实现深拷贝的功能,值得注意的是:
// 第二版
function extend() {
// 默认不进行深拷贝
var deep = false;
var name, options, src, copy;
var length = arguments.length;
// 记录要复制的对象的下标
var i = 1;
// 第一个参数不传布尔值的状况下,target默认是第一个参数
var target = arguments[0] || {};
// 若是第一个参数是布尔值,第二个参数是才是target
if (typeof target == 'boolean') {
deep = target;
target = arguments[i] || {};
i++;
}
// 若是target不是对象,咱们是没法进行复制的,因此设为{}
if (typeof target !== 'object') {
target = {}
}
// 循环遍历要复制的对象们
for (; i < length; i++) {
// 获取当前对象
options = arguments[i];
// 要求不能为空 避免extend(a,,b)这种状况
if (options != null) {
for (name in options) {
// 目标属性值
src = target[name];
// 要复制的对象的属性值
copy = options[name];
if (deep && copy && typeof copy == 'object') {
// 递归调用
target[name] = extend(deep, src, copy);
}
else if (copy !== undefined){
target[name] = copy;
}
}
}
}
return target;
};复制代码
在实现上,核心的部分仍是跟上篇实现的深浅拷贝函数一致,若是要复制的对象的属性值是一个对象,就递归调用 extend。不过 extend 的实现中,多了不少细节上的判断,好比第一个参数是不是布尔值,target 是不是一个对象,不传参数时的默认值等。
接下来,咱们看几个 jQuery 的 extend 使用效果:
在咱们的实现中,typeof target
必须等于 object
,咱们才会在这个 target
基础上进行拓展,然而咱们用 typeof
判断一个函数时,会返回function
,也就是说,咱们没法在一个函数上进行拓展!
什么,咱们还能在一个函数上进行拓展!!
固然啦,毕竟函数也是一种对象嘛,让咱们看个例子:
function a() {}
a.target = 'b';
console.log(a.target); // b复制代码
实际上,在 underscore 的实现中,underscore 的各类方法即是挂在了函数上!
因此在这里咱们还要判断是否是函数,这时候咱们即可以使用《JavaScript专题之类型判断(上)》中写得 isFunction 函数
咱们这样修改:
if (typeof target !== "object" && !isFunction(target)) {
target = {};
}复制代码
其实咱们实现的方法有个小 bug ,不信咱们写个 demo:
var obj1 = {
a: 1,
b: {
c: 2
}
}
var obj2 = {
b: {
c: [5],
}
}
var d = extend(true, obj1, obj2)
console.log(d);复制代码
咱们预期会返回这样一个对象:
{
a: 1,
b: {
c: [5]
}
}复制代码
然而返回了这样一个对象:
{
a: 1,
b: {
c: {
0: 5
}
}
}复制代码
让咱们细细分析为何会致使这种状况:
首先咱们在函数的开始写一个 console 函数好比:console.log(1),而后以上面这个 demo 为例,执行一下,咱们会发现 1 打印了三次,这就是说 extend 函数执行了三遍,让咱们捋一捋这三遍传入的参数:
第一遍执行到递归调用时:
var src = { c: 2 };
var copy = { c: [5]};
target[name] = extend(true, src, copy);复制代码
第二遍执行到递归调用时:
var src = 2;
var copy = [5];
target[name] = extend(true, src, copy);复制代码
第三遍进行最终的赋值,由于 src 是一个基本类型,咱们默认使用一个空对象做为目标值,因此最终的结果就变成了对象的属性!
为了解决这个问题,咱们须要对目标属性值和待复制对象的属性值进行判断:
判断目标属性值跟要复制的对象的属性值类型是否一致:
若是待复制对象属性值类型为数组,目标属性值类型不为数组的话,目标属性值就设为 []
若是待复制对象属性值类型为对象,目标属性值类型不为对象的话,目标属性值就设为 {}
结合着《JavaScript专题之类型判断(下)》中的 isPlainObject 函数,咱们能够对类型进行更细致的划分:
var clone, copyIsArray;
...
if (deep && copy && (isPlainObject(copy) ||
(copyIsArray = Array.isArray(copy)))) {
if (copyIsArray) {
copyIsArray = false;
clone = src && Array.isArray(src) ? src : [];
} else {
clone = src && isPlainObject(src) ? src : {};
}
target[name] = extend(deep, clone, copy);
} else if (copy !== undefined) {
target[name] = copy;
}复制代码
实际上,咱们还可能遇到一个循环引用的问题,举个例子:
var a = {name : b};
var b = {name : a}
var c = extend(a, b);
console.log(c);复制代码
咱们会获得一个能够无限展开的对象,相似于这样:
为了不这个问题,咱们须要判断要复制的对象属性是否等于 target,若是等于,咱们就跳过:
...
src = target[name];
copy = options[name];
if (target === copy) {
continue;
}
...复制代码
若是加上这句,结果就会是:
{name: undefined}复制代码
function extend() {
// 默认不进行深拷贝
var deep = false;
var name, options, src, copy, clone, copyIsArray;
var length = arguments.length;
// 记录要复制的对象的下标
var i = 1;
// 第一个参数不传布尔值的状况下,target 默认是第一个参数
var target = arguments[0] || {};
// 若是第一个参数是布尔值,第二个参数是 target
if (typeof target == 'boolean') {
deep = target;
target = arguments[i] || {};
i++;
}
// 若是target不是对象,咱们是没法进行复制的,因此设为 {}
if (typeof target !== "object" && !isFunction(target)) {
target = {};
}
// 循环遍历要复制的对象们
for (; i < length; i++) {
// 获取当前对象
options = arguments[i];
// 要求不能为空 避免 extend(a,,b) 这种状况
if (options != null) {
for (name in options) {
// 目标属性值
src = target[name];
// 要复制的对象的属性值
copy = options[name];
// 解决循环引用
if (target === copy) {
continue;
}
// 要递归的对象必须是 plainObject 或者数组
if (deep && copy && (isPlainObject(copy) ||
(copyIsArray = Array.isArray(copy)))) {
// 要复制的对象属性值类型须要与目标属性值相同
if (copyIsArray) {
copyIsArray = false;
clone = src && Array.isArray(src) ? src : [];
} else {
clone = src && isPlainObject(src) ? src : {};
}
target[name] = extend(deep, clone, copy);
} else if (copy !== undefined) {
target[name] = copy;
}
}
}
}
return target;
};复制代码
若是以为看明白了上面的代码,想一想下面两个 demo 的结果:
var a = extend(true, [4, 5, 6, 7, 8, 9], [1, 2, 3]);
console.log(a) // ???复制代码
var obj1 = {
value: {
3: 1
}
}
var obj2 = {
value: [5, 6, 7],
}
var b = extend(true, obj1, obj2) // ???
var c = extend(true, obj2, obj1) // ???复制代码
JavaScript专题系列目录地址:github.com/mqyqingfeng…。
JavaScript专题系列预计写二十篇左右,主要研究平常开发中一些功能点的实现,好比防抖、节流、去重、类型判断、拷贝、最值、扁平、柯里、递归、乱序、排序等,特色是研(chao)究(xi) underscore 和 jQuery 的实现方式。
若是有错误或者不严谨的地方,请务必给予指正,十分感谢。若是喜欢或者有所启发,欢迎 star,对做者也是一种鼓励。