3

I have this array with objects:

var data=[
    { number: '31907411282', unix_date: '1547012427' },
    { number: '31907411282', unix_date: '1547013214' },
    { number: '31907514691', unix_date: '1547015155' },
    { number: '31907514691', unix_date: '1547015232' },
    { number: '31907514691', unix_date: '1547016134' },
    { number: '31907514691', unix_date: '1547016443' },
    { number: '31907638716', unix_date: '1547017122' },
    { number: '31907644067', unix_date: '1547017381' }
]

How I can remove duplicate objects with newer date and get new array like this:

var new_data=[
    { number: '31907411282', unix_date: '1547012427' },
    { number: '31907514691', unix_date: '1547015155' },
    { number: '31907638716', unix_date: '1547017122' },
    { number: '31907644067', unix_date: '1547017381' }
]

I can't understand how to do all procedure. Thanks

UPDATE

I try this construction:

  1. firstly we need group and sort objects
  2. second step is get first object from grouped and push to new array

Code:

var data=[
    { number: '31907411282', unix_date: '1547012427' },
    { number: '31907411282', unix_date: '1547013214' },
    { number: '31907514691', unix_date: '1547019134' },
    { number: '31907514691', unix_date: '1547015232' },
    { number: '31907514691', unix_date: '1547016134' },
    { number: '31907514691', unix_date: '1547016443' },
    { number: '31907638716', unix_date: '1547017122' },
    { number: '31907644067', unix_date: '1547017381' }
]

var new_data=[];
var groupByNumber=groupBy(data, 'number');
for(i in groupByNumber){
    var sort_arr=groupByNumber[i].sort(compare)
    new_data.push(sort_arr[0])
}

console.log(new_data)

function compare(a, b) {
  let comparison = 0;
  if (a.unix_date > b.unix_date) {
    comparison = 1;
  } else if (a.unix_date < b.unix_date) {
    comparison = -1;
  }
  return comparison;
}

function groupBy(xs, key) { //group by key
    return xs.reduce(function(rv, x) {
        (rv[x[key]] = rv[x[key]] || []).push(x);
        return rv;
    }, {});
};

Maybe it can be more elegant

UPDATE 2

@miroslav-glamuzina give more elegance way:

let filtered = data.reduce((acc, item) => {
  if (!acc.some((e, i) => {
      if (item.number === e.number) {
        if (item.unix_date < e.unix_date) {
          acc.splice(i, 1, item);
        }
        return true;
      }
    })) {
    acc.push(item);
  }
  return acc;
}, []);
1
  • Sort the array by date. Then go through the array, adding elements to a result array if the number isn't already in the result. Commented Mar 18, 2019 at 1:10

3 Answers 3

2

You can use reduce to loop through the array and store a set of keys(number) to check if this exists. Then push into the new array.

var data=[
    { number: '31907411282', unix_date: '1547012427' },
    { number: '31907411282', unix_date: '1547013214' },
    { number: '31907514691', unix_date: '1547015155' },
    { number: '31907514691', unix_date: '1547015232' },
    { number: '31907514691', unix_date: '1547016134' },
    { number: '31907514691', unix_date: '1547016443' },
    { number: '31907638716', unix_date: '1547017122' },
    { number: '31907644067', unix_date: '1547017381' }
]
const rs = data.reduce((acc, e) => {
  if(!acc.number.has(e.number)) {
    acc.newData.push(e)
    acc.number.add(e.number)
  }
  return acc
}, {number: new Set(), newData: []})

console.log(rs.newData)

Sign up to request clarification or add additional context in comments.

2 Comments

Very nice use of Set's! Although, if the data in the array is not ordered, this code will not return the max unix_date of each number :/
Because the OP's data is already sorted then I think we don't have to do that. My bad.
1

You can use reduce to achieve this:

var data = [{
    number: '31907411282',
    unix_date: '1547013214'
  },
  {
    number: '31907411282',
    unix_date: '1547012427'
  },
  {
    number: '31907514691',
    unix_date: '1547015155'
  },
  {
    number: '31907514691',
    unix_date: '1547015232'
  },
  {
    number: '31907514691',
    unix_date: '1547016134'
  },
  {
    number: '31907514691',
    unix_date: '1547016443'
  },
  {
    number: '31907638716',
    unix_date: '1547017122'
  },
  {
    number: '31907644067',
    unix_date: '1547017381'
  }
];

let filtered = data.reduce((acc, item) => {
  if (!acc.some((e, i) => {
      if (item.number === e.number) {
        if (item.unix_date < e.unix_date) {
          acc.splice(i, 1, item);
        }
        return true;
      }
    })) {
    acc.push(item);
  }
  return acc;
}, []);

console.log(filtered);

Hope this helps,

Comments

1

Sort the items by date, reduce the array, keeping only the elements that have a number that does not already exist in the accumulator array.

let data = [{
      number: '31907411282',
      unix_date: '1547013214'
    },
    {
      number: '31907411282',
      unix_date: '1547012427'
    },
    {
      number: '31907514691',
      unix_date: '1547015155'
    },
    {
      number: '31907514691',
      unix_date: '1547015232'
    },
    {
      number: '31907514691',
      unix_date: '1547016134'
    },
    {
      number: '31907514691',
      unix_date: '1547016443'
    },
    {
      number: '31907638716',
      unix_date: '1547017122'
    },
    {
      number: '31907644067',
      unix_date: '1547017381'
    }
    ];

    let filtered = data
      .sort((a, b) => (a.unix_date > b.unix_date) ? 1 : -1)
      .reduce((acc, item) => {
        if (!acc.find(el => el.number === item.number)) {
          acc.push(item)
        }
        return acc
      }, []);

      console.log(filtered);

Comments

Your Answer

By clicking “Post Your Answer”, you agree to our terms of service and acknowledge you have read our privacy policy.

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.