我正在处理NodeJS中对象的深拷贝问题。我的自定义扩展功能不太好用。Underscore的扩展功能是平的。虽然在stackexchange上有一些相当简单的扩展变体,但没有一个接近jQuery.extend(true,{},obj,obj,obj)..(大多数实际上很糟糕,并且会破坏异步代码的优势)。因此,我的问题是:是否有适用于NodeJS的良好深度副本?有人移植了jQuery的吗?
我正在处理NodeJS中对象的深拷贝问题。我的自定义扩展功能不太好用。Underscore的扩展功能是平的。虽然在stackexchange上有一些相当简单的扩展变体,但没有一个接近jQuery.extend(true,{},obj,obj,obj)..(大多数实际上很糟糕,并且会破坏异步代码的优势)。因此,我的问题是:是否有适用于NodeJS的良好深度副本?有人移植了jQuery的吗?
已经进行了移植。node-extend
请注意,该项目没有测试并且不太受欢迎,因此自行承担风险。
如上所述,您可能不需要进行深度复制。尝试更改数据结构,使您仅需要浅层复制。
相反,我撰写了一个较小的模块,建议您使用xtend。它没有包含jQuery的实现,也没有像node-extend那样存在错误。
你需要jQuery,那就直接使用它:
function extend() {
var options, name, src, copy, copyIsArray, clone, target = arguments[0] || {},
i = 1,
length = arguments.length,
deep = false,
toString = Object.prototype.toString,
hasOwn = Object.prototype.hasOwnProperty,
push = Array.prototype.push,
slice = Array.prototype.slice,
trim = String.prototype.trim,
indexOf = Array.prototype.indexOf,
class2type = {
"[object Boolean]": "boolean",
"[object Number]": "number",
"[object String]": "string",
"[object Function]": "function",
"[object Array]": "array",
"[object Date]": "date",
"[object RegExp]": "regexp",
"[object Object]": "object"
},
jQuery = {
isFunction: function (obj) {
return jQuery.type(obj) === "function"
},
isArray: Array.isArray ||
function (obj) {
return jQuery.type(obj) === "array"
},
isWindow: function (obj) {
return obj != null && obj == obj.window
},
isNumeric: function (obj) {
return !isNaN(parseFloat(obj)) && isFinite(obj)
},
type: function (obj) {
return obj == null ? String(obj) : class2type[toString.call(obj)] || "object"
},
isPlainObject: function (obj) {
if (!obj || jQuery.type(obj) !== "object" || obj.nodeType) {
return false
}
try {
if (obj.constructor && !hasOwn.call(obj, "constructor") && !hasOwn.call(obj.constructor.prototype, "isPrototypeOf")) {
return false
}
} catch (e) {
return false
}
var key;
for (key in obj) {}
return key === undefined || hasOwn.call(obj, key)
}
};
if (typeof target === "boolean") {
deep = target;
target = arguments[1] || {};
i = 2;
}
if (typeof target !== "object" && !jQuery.isFunction(target)) {
target = {}
}
if (length === i) {
target = this;
--i;
}
for (i; i < length; i++) {
if ((options = arguments[i]) != null) {
for (name in options) {
src = target[name];
copy = options[name];
if (target === copy) {
continue
}
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 : {};
}
// WARNING: RECURSION
target[name] = extend(deep, clone, copy);
} else if (copy !== undefined) {
target[name] = copy;
}
}
}
}
return target;
}
并做一个小测试以展示它确实进行了深拷贝
extend(true,
{
"name": "value"
},
{
"object": "value",
"other": "thing",
"inception": {
"deeper": "deeper",
"inception": {
"deeper": "deeper",
"inception": {
"deeper": "deeper"
}
}
}
}
)
但是要记得提供归属权:https://github.com/jquery/jquery/blob/master/src/core.js
请使用内置的util模块:
var extend = require('util')._extend;
var merged = extend(obj1, obj2);
util._extend
不是深层复制。 - dbkaplun快速且简单的深拷贝方法是使用一些 JSON 技巧。虽然这不是最高效的方法,但它确实可以非常好地完成工作。
function clone(a) {
return JSON.parse(JSON.stringify(a));
}
它只适用于复制简单的值对象;对于日期、函数和某些实例化对象,它会失败。
- Kato我知道这是一个老问题,但我想推荐 lodash的合并函数 作为一个好的解决方案。我总体上建议使用lodash来处理实用函数 :)
extend
会改变对象,这太糟糕了。 - Wtowermerge
和extend
的第一个参数中使用空对象来轻松避免。
var obj3 = lodash.extend(obj1, obj2)
会改变obj1
var obj3 = lodash.extend({}, obj1, obj2)
不会改变obj1
。 - edgar.bjorntvedt这适用于深度对象扩展...请注意,它会取代数组而不是它们的值,但显然可以根据你的喜好进行更新。它应该保持枚举功能和您可能希望它执行的所有其他操作
function extend(dest, from) {
var props = Object.getOwnPropertyNames(from), destination;
props.forEach(function (name) {
if (typeof from[name] === 'object') {
if (typeof dest[name] !== 'object') {
dest[name] = {}
}
extend(dest[name],from[name]);
} else {
destination = Object.getOwnPropertyDescriptor(from, name);
Object.defineProperty(dest, name, destination);
}
});
}
_.extend = extendify({
inPlace: false,
isDeep: true
});
只需安装扩展包。 文档: node扩展包 安装:
npm install extend
extend ( [deep], target, object1, [objectN] )
deep是可选的。默认为false。如果切换为true,则会递归合并您的对象。