Reputation: 1925
I'd like to use the _.union
function to create a union of two arrays of objects. Union works with arrays of primitives only as it uses === to examine if two values are equal.
I'd like to compare objects using a key property: objects with the same key property would be regarded equal. Is there a nice functional way to achieve that ideally using lodash?
Upvotes: 18
Views: 48352
Reputation: 2299
_.spread(_.merge)
would do the trick.
_.spread(_.merge)([{ a: 1 }, { b: 2 }, { c: 3 }])
would return { a: 1, b: 2, c: 3 }
Upvotes: 1
Reputation: 341
And what about UniqBy with a concat of the two arrays before?
import _ from 'lodash'
const arrayUnion = (arr1, arr2, identifier) => {
const array = [...arr1, ...arr2]
return _.uniqBy(array, identifier)
}
const array1 = [{ id: 1 }, { id: 2 }, { id: 3 }]
const array2 = [{ id: 3 }, { id: 4 }, { id: 4 }]
console.log(arrayUnion(array1, array2, 'id'))
result → [{ 'id': 1 }, { 'id': 2 }, { 'id': 3 }, { 'id': 4 }]
Upvotes: 18
Reputation: 4384
_.unionBy(array1,array2,'propname');
_.unionWith(array1,array2, _.isEqual);
_.unionWith(array1,array2, function(obj,other){ return obj.propname.toLowercase() === other.propname.toLowercase(); });
propname is name of key property of your object.
Upvotes: 1
Reputation: 3423
Late to the party but _.unionWith is much better in doing what you want.
var objects = [{ 'x': 1, 'y': 2 }, { 'x': 2, 'y': 1 }];
var others = [{ 'x': 1, 'y': 1 }, { 'x': 1, 'y': 2 }];
_.unionWith(objects, others, _.isEqual);
// => [{ 'x': 1, 'y': 2 }, { 'x': 2, 'y': 1 }, { 'x': 1, 'y': 1 }]
Upvotes: 16
Reputation: 2485
lodash merge object from arrays
const test1 = [
{ name: 'zhanghong', age: 32, money: 0, size: 12, },
{ name: 'wanghong', age: 20, size: 6 },
{ name: 'jinhong', age: 16, height: 172 },
]
const test2 = [
{ name: 'zhanghong', gender: 'male', age: 14 },
{ name: 'wanghong', gender: 'female', age: 33 },
{ name: 'lihong', gender: 'female', age: 33 },
]
const test3 = [
{ name: 'meinv' },
]
const test4 = [
{ name: 'aaa' },
]
const test5 = [
{ name: 'zhanghong', age: 'wtf' },
]
const result = mergeUnionByKey(test1, test2, test3, test4, [], test5, 'name', 'override')
function mergeUnionByKey(...args) {
const config = _.chain(args)
.filter(_.isString)
.value()
const key = _.get(config, '[0]')
const strategy = _.get(config, '[1]') === 'override' ? _.merge : _.defaultsDeep
if (!_.isString(key))
throw new Error('missing key')
const datasets = _.chain(args)
.reject(_.isEmpty)
.filter(_.isArray)
.value()
const datasetsIndex = _.mapValues(datasets, dataset => _.keyBy(dataset, key))
const uniqKeys = _.chain(datasets)
.flatten()
.map(key)
.uniq()
.value()
return _.chain(uniqKeys)
.map(val => {
const data = {}
_.each(datasetsIndex, dataset => strategy(data, dataset[val]))
return data
})
.filter(key)
.value()
}
console.log(JSON.stringify(result, null, 4))
<script src="https://cdnjs.cloudflare.com/ajax/libs/lodash.js/4.17.4/lodash.min.js"></script>
Upvotes: 2
Reputation: 10834
A non pure lodash way to do this but using the array.concat function you are able to do this pretty simply along uniq()
:
var objUnion = function(array1, array2, matcher) {
var concated = array1.concat(array2)
return _.uniq(concated, false, matcher);
}
An alternative approach would be to use flatten() and uniq():
var union = _.uniq(_.flatten([array1, array2]), matcherFn);
Upvotes: 17