7

EDIT: not a duplicate, as my question does not ask for deleting a specific key, but all keys not found in an array.

In the below, function redux1 removes entries corresponding to keys not listed in keys_to_keep from data object.

Given that I have a list of object keys to keep, how can I rewrite redux1 in a cleaner way, prefarably utilizing map, filter or reduce?

var data = [
 {name: 'John', city: 'London', age: 42},
 {name: 'Mike', city: 'Warsaw', age: 18},
 {name: 'Jim', city: 'New York', age: 22},
 {name: 'Celine', city: 'Tokyo', age: 54},
]

var keys_to_keep = ['name', 'city']

function redux1(data) {
 data.forEach((person) => {
  Object.keys(person).forEach((key) => {
   if (!keys_to_keep.includes(key)) {
    delete (person[key])
   }
  })
 })
 console.log(data)
}

function redux2(data) { 
 var reduced = data.filter(person => Object.keys(person).filter(key => keys_to_keep.includes(key)))
 console.log(reduced)
}

redux1(data)
//redux2(data)

My current redux2 will return objects will not remove age.

barciewicz
  • 2,865
  • 4
  • 25
  • 58
  • _"...in a cleaner way"_ - what's not "clean" about your function? _"...prefarably utilizing map, filter or reduce?"_ - Why those three? – Andreas Feb 27 '19 at 14:25
  • @Heretic Monkey, not a duplicate. I know this answer, but it only covers deleting specific key, not all keys not found in an array. – barciewicz Feb 27 '19 at 14:53
  • Then you should mention that in your question. https://stackoverflow.com/help/duplicates – Heretic Monkey Feb 27 '19 at 14:54

6 Answers6

21

You could use a combination of Array#map and Array#reduce:

const data = [
 {name: 'John', city: 'London', age: 42},
 {name: 'Mike', city: 'Warsaw', age: 18},
 {name: 'Jim', city: 'New York', age: 22},
 {name: 'Celine', city: 'Tokyo', age: 54},
]

const keys_to_keep = ['name', 'city'];

const redux = array => array.map(o => keys_to_keep.reduce((acc, curr) => {
  acc[curr] = o[curr];
  return acc;
}, {}));

console.log(redux(data));
Zenoo
  • 12,242
  • 4
  • 44
  • 63
3

A version slightly shorter than the accepted answer using Array#map and Object.fromEntries():

const data = [
    {name: 'John', city: 'London', age: 42},
    {name: 'Mike', city: 'Warsaw', age: 18},
    {name: 'Jim', city: 'New York', age: 22},
    {name: 'Celine', city: 'Tokyo', age: 54},
]

const keys_to_keep = ['name', 'city'];

const redux1 = list => list.map(o => Object.fromEntries(
    keys_to_keep.map(k => [k, o[k]])
));

console.log(redux1(data));
zr0gravity7
  • 2,308
  • 1
  • 6
  • 22
Dragonfang
  • 304
  • 1
  • 7
2

use Array.map and Array.forEach inside it :

var data = [
 {name: 'John', city: 'London', age: 42},
 {name: 'Mike', city: 'Warsaw', age: 18},
 {name: 'Jim', city: 'New York', age: 22},
 {name: 'Celine', city: 'Tokyo', age: 54},
]

var keys_to_keep = ['name', 'city']

const result = data.map(e => {
  const obj = {};
  keys_to_keep.forEach(k => obj[k] = e[k])
  return obj;
});

console.log(result);
Taki
  • 16,417
  • 3
  • 24
  • 44
1

var data = [
 {name: 'John', city: 'London', age: 42},
 {name: 'Mike', city: 'Warsaw', age: 18},
 {name: 'Jim', city: 'New York', age: 22},
 {name: 'Celine', city: 'Tokyo', age: 54},
]

var keys_to_keep = ['name', 'city']

data=data.map(element => Object.assign({}, ...keys_to_keep.map(key => ({[key]: element[key]}))))

console.log(data)
Krzysztof Krzeszewski
  • 5,299
  • 2
  • 14
  • 27
0
data.reduce((r, c) => [ ...r, Object.entries(c).reduce((b, [k, v]) => keys_to_keep.includes(k) ? {...b, [k]: v } : b, {}) ],[])
A. Moynet
  • 410
  • 3
  • 8
0

You can use Object.entries

function objfilter(data,keys_to_keep){
    return Object.fromEntries(Object.entries(data).filter(a=>keys_to_keep.includes(a[0])))
}
const data = [
    {name: 'John', city: 'London', age: 42},
    {name: 'Mike', city: 'Warsaw', age: 18},
    {name: 'Jim', city: 'New York', age: 22},
    {name: 'Celine', city: 'Tokyo', age: 54},
]

const keys_to_keep = ['name', 'city'];

console.log(objfilter(data,keys_to_keep))
Rainb
  • 1,520
  • 8
  • 28