如何使用下划线克隆对象数组?


82
#!/usr/bin/env node
var _ = require('underscore');
var a = [{f: 1}, {f:5}, {f:10}];
var b = _.clone(a);
b[1].f = 55;
console.log(JSON.stringify(a));

结果是:

[{"f":1},{"f":55},{"f":10}]

克隆似乎无效!因此,我进行RTFM,然后看到以下内容:

http://underscorejs.org/#clone

创建对象的浅表副本。任何嵌套的对象或数组都将通过引用复制,而不是重复。

所以_.clone是非常没用的。有没有一种方法可以实际复制对象数组?


4
有人请求复制深层副本的请求被拒绝:github.com/jashkenas/underscore/pull/595 Lo-Dash具有cloneDeep
epascarello,2014年

5
大声笑我刚刚意识到下划线的双关语。低破折号。
杰西2014年

Answers:


122

好吧,有个把戏!如果clone不“克隆”嵌套对象,则可以通过显式克隆map调用中的每个对象来强制它!像这样:

#!/usr/bin/env node
var _ = require('underscore');
var a = [{f: 1}, {f:5}, {f:10}];
var b = _.map(a, _.clone);       // <----
b[1].f = 55;
console.log(JSON.stringify(a));

印刷品:

[{"f":1},{"f":5},{"f":10}]

好极了! a不变!我现在可以b按自己的喜好编辑!


48
不过要小心。当然,这仅适用于两个层次。不适用于嵌套比该示例更多的数组或对象。
Simon Zyx

1
根据设计,Underscore也不会正确克隆RegExp或Date值
Mark K Cowan

1
任何人在这里都应该在下面看到我的答案。
gdibble '17

65

从Github的问题中提取的另一种解决方案,可以使用任何级别的嵌套数据,并且不需要下划线:

JSON.parse(JSON.stringify(obj))

13
除非对象具有循环(在这种情况下JSON.stringify会引发错误),否则此方法有效。原始情况并非如此,但仍然是一种有趣的状态。 a = {simple: 'thing'}; a.cycle = a ; JSON.stringify(a)
mcdave

10
同样值得注意的是,该解决方案仅适用于具有简单类型的对象。例如,如果您是对象拥有DateRegex实例,则它们将被序列化为字符串。不是世界末日,但是如果您正在使用它并期望有Date实例,则需要处理它。
cayleyh 2015年

1
而且,如果您认为有人可能会尝试undefined执行此操作,那么您会想要JSON.parse(JSON.stringify(obj) || null)否则将引发错误。
伊恩·麦金农

1
连同@cayleyh所提到的一起,这将function完全消失。
MarkoGrešak16年


9

下划线API参考

_.toArray(list)从列表(任何可以迭代的对象)创建一个实际的Array。对于转换arguments对象很有用。

...或者在这种情况下,克隆数组。试试这个:

var _ = require('underscore');
var array1 =  [{a:{b:{c:1}}},{b:{c:{a:2}}},{c:{a:{b:3}}}];
var array2 = _.toArray(array1);
console.log(array1 === array2); --> false
console.log(array1[0] === array2[0]); --> true

以下是我在史蒂夫的评论-thx之后创建的附录

Vanilla JS(或_.clone根据需要使用)深度克隆递归帮助器

function clone(thing, opts) {
    var newObject = {};
    if (thing instanceof Array) {
        return thing.map(function (i) { return clone(i, opts); });
    } else if (thing instanceof Date) {
        return new Date(thing);
    } else if (thing instanceof RegExp) {
        return new RegExp(thing);
    } else if (thing instanceof Function) {
        return opts && opts.newFns ?
                   new Function('return ' + thing.toString())() :
                   thing;
    } else if (thing instanceof Object) {
        Object.keys(thing).forEach(function (key) {
            newObject[key] = clone(thing[key], opts);
        });
        return newObject;
    } else if ([ undefined, null ].indexOf(thing) > -1) {
        return thing;
    } else {
        if (thing.constructor.name === 'Symbol') {
            return Symbol(thing.toString()
                       .replace(/^Symbol\(/, '')
                       .slice(0, -1));
        }
        // return _.clone(thing);  // If you must use _ ;)
        return thing.__proto__.constructor(thing);
    }
}

var a = {
    a: undefined,
    b: null,
    c: 'a',
    d: 0,
    e: Symbol('a'),
    f: {},
    g: { a:1 },
    h: [],
    i: [ { a:2 }, { a:3 } ],
    j: [ 1, 2 ],
    k: function (a) { return a; },
    l: /[a-z]/g,
    z: [ {
        a: undefined,
        b: null,
        c: 'b',
        d: 1,
        e: Symbol(1),
        f: {},
        g: { b:2 },
        h: { c:{ c:3 } },
        i: { a:Symbol('b') },
        j: { a:undefined, b:null },
        k: [],
        l: [ 1, [ 1, 2 ], [ [ 1, 2, 3 ] ] ],
        m: function (a) { return !a; },
        n: { a:function (a) { return !!a; } },
        o: /(a|b)/i
       } ]
};
var b = clone(a);
var c = clone(a, { newFns:true });


/* Results - value beneath each for reference:

a.a === b.a --> true
undefined

a.b === b.b --> true
null

a.c === b.c --> true
'a'

a.d === b.d --> true
0

a.e === b.e --> false
Symbol(a)

a.f === b.f --> false
{}

a.g === b.g --> false
{ a:1 }

a.h === b.h --> false
[]

a.i === b.i --> false
[ { a:2 }, { a:3 } ]

a.i[0] === b.i[0] --> false
{ a:2 }

a.i[0].a === b.i[0].a --> true
2

a.j === b.j --> false
[ 1, 2 ]

a.k === b.k --> true
a.k === c.k --> false
function (a) { return a; }

a.l === b.l --> false
/[a-z]/g

a.z === b.z --> false
[Object]

a.z[0].a === b.z[0].a --> true
undefined

a.z[0].b === b.z[0].b --> true
null

a.z[0].c === b.z[0].c --> true
'b'

a.z[0].d === b.z[0].d --> true
1

a.z[0].e === b.z[0].e --> 
false
Symbol(1)

a.z[0].f === b.z[0].f --> false
{}

a.z[0].g === b.z[0].g -- > false
{ b:2 }

a.z[0].g.b === b.z[0].g.b --> true
2

a.z[0].h === b.z[0].h --> false
{ c:{ c:3 } }

a.z[0].h.c === b.z[0].h.c --> false
{ c:3 }

a.z[0].h.c.c === b.z[0].h.c.c --> true
3

a.z[0].i === b.z[0].i --> false
{ a:Symbol(b) }

a.z[0].i.a === b.z[0].i.a --> false
Symbol(b)

a.z[0].j === b.z[0].j --> false
{ a:undefined, b:null }

a.z[0].j.a === b.z[0].j.a --> true
undefined

a.z[0].k === b.z[0].k --> false
[]

a.z[0].l === b.z[0].l --> false
[ 1, [ 1, 2 ], [ [ 1, 2, 3 ] ] ]

a.z[0].l[1] === b.z[0].l[1] --> false
[ 1, 2 ]

a.z[0].l[1][1] === b.z[0].l[1][1] --> true
2

a.z[0].m === b.z[0].m --> true
a.z[0].m === c.z[0].m --> false
function (a) { return !a; }

a.z[0].n === b.z[0].n --> false
{ a:function (a) { return !!a; } }

a.z[0].n.a === b.z[0].n.a --> true
a.z[0].n.a === c.z[0].n.a --> false
function (a) { return !!a; }

a.z[0].o === b.z[0].o --> false
/(a|b)/i

*/

这是最好的答案。
Pierre

_.toArray(list)不克隆数组中的对象。 var array1 = [{a: 1}, {a: 2}, {a: 3}]; var array2 = _.toArray(array1); array2[0].a = 999; console.log(array1[0]); --> {a: 999}
史蒂夫·朗

@SteveLang感谢您指出这一点。哎呀。因此,我花了一些时间来制作上面的普通JS fn,如果用户真的必须可以_.cloneelse条件下使用;那么
gdibble
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.