Home > database >  Find missing months in js array
Find missing months in js array

Time:05-18

I have the following Array

[
  { Month: '2021-05', Count: 36 },
  { Month: '2021-06', Count: 1048 },
  { Month: '2021-07', Count: 572 },
  { Month: '2021-09', Count: 3 },
  { Month: '2021-12', Count: 52 },
  { Month: '2022-01', Count: 4 },
  { Month: '2022-02', Count: 273 },
  { Month: '2022-04', Count: 96 }
]

where I am missing a few months. I know how many months is needed (could be 12 or could be more or less) and I need the missing months (like 2021-08 in this case) to be added with a count of 0. How to go about it?

CodePudding user response:

Here's a pure, functional approach which will create a new array with new items, inserting all of the missing months in order. The code includes some comments explaining the procedure:

const parseDate = str => str.split('-').map(Number);
const formatDate = (year, month) => `${year}-${String(month).padStart(2, '0')}`;

function createContinuousMonthCounts (array) {
  const all = [];
  // get initial year/month values from first item
  let [year, month] = parseDate(array[0].Month);

  const advanceDate = () => {
    month  = 1;
    if (month > 12) {
      year  = 1;
      month = 1;
    }
  };

  for (const item of array) {
    const [y, m] = parseDate(item.Month);
    // while the current month is not equal to the current item's month,
    // create an entry for the month, append it, and advance to the next month
    while (year !== y || month !== m) {
      all.push({Month: formatDate(year, month), Count: 0});
      advanceDate();
    }
    // after we're up to date, add the current item and advance the date
    all.push({...item});
    advanceDate();
  }

  return all;
}

const array = [
  { Month: '2021-05', Count: 36 },
  { Month: '2021-06', Count: 1048 },
  { Month: '2021-07', Count: 572 },
  { Month: '2021-09', Count: 3 },
  { Month: '2021-12', Count: 52 },
  { Month: '2022-01', Count: 4 },
  { Month: '2022-02', Count: 273 },
  { Month: '2022-04', Count: 96 },
];

const all = createContinuousMonthCounts(array);

for (const {Month, Count} of all) console.log(Month, Count);

CodePudding user response:

Just a shot into the dark (please consider adding some Code to your question):

const months = [
  { Month: '2021-05', Count: 36 },
  { Month: '2021-06', Count: 1048 },
  { Month: '2021-07', Count: 572 },
  { Month: '2021-09', Count: 3 },
  { Month: '2021-12', Count: 52 },
  { Month: '2022-01', Count: 4 },
  { Month: '2022-02', Count: 273 },
  { Month: '2022-04', Count: 96 }
];

const neededMonths = [
  "2021-01","2021-02","2021-03","2021-04","2021-05","2021-06","2021-07","2021-08","2021-09","2021-10","2021-11","2021-12"
]

const missedMonths = [];

months.map( m => {
  if(neededMonths.indexOf(m.Month) == -1 ){
    missedMonths.push(m.Month);
  }
});

console.log(missedMonths);

CodePudding user response:

You first need a method to find all the months between a range, then iterate across all the months and add the missing ones with count: 0:

const months = [
  { Month: '2021-05', Count: 36 },
  { Month: '2021-06', Count: 1048 },
  { Month: '2021-07', Count: 572 },
  { Month: '2021-09', Count: 3 },
  { Month: '2021-12', Count: 52 },
  { Month: '2022-01', Count: 4 },
  { Month: '2022-02', Count: 273 },
  { Month: '2022-04', Count: 96 }
]

const firstMonth = months.at(0).Month;
const lastMonth = months.at(-1).Month;

const [initialYear, initialMonth] = firstMonth.split('-');
const [endingYear, endingMonth] = lastMonth.split('-');

const allMonths = [];

let currentMonth = initialMonth;
let currentYear = initialYear;
while (`${currentYear}-${('' currentMonth).padStart(2, '0')}` !== lastMonth) {
  allMonths.push(`${currentYear}-${('' currentMonth).padStart(2, '0')}`);
  currentMonth  ;
  if (currentMonth === 13) {
    currentMonth = 1;
    currentYear  ;
  }
}

allMonths.forEach(month => {
  if (!months.find(m => m.Month === month)) {
     months.push({Month: month, count: 0});
  }
});

console.log(months);

CodePudding user response:

Here is a simple solution:

  • Sort the input array by date and iterate
  • Insert each item in a new array called result
    • But before inserting, check if last month in result array and current month have gaps; use a nested loop to fill the gap

const array = [
  { Month: "2021-05", Count: 36 },
  { Month: "2021-06", Count: 1048 },
  { Month: "2021-07", Count: 572 },
  { Month: "2021-09", Count: 3 },
  { Month: "2021-12", Count: 52 },
  { Month: "2022-01", Count: 4 },
  { Month: "2022-02", Count: 273 },
  { Month: "2022-04", Count: 96 }
];
let result = [];
array.sort(function(a, b) {
  return a.Month.localeCompare(b.Month);
});
array.forEach(function(item) {
  function addmonth(yymm) {
    let yy = Number(yymm.split("-")[0]);
    let mm = Number(yymm.split("-")[1])   1;
    mm  = 1;
    if (mm > 12) {
      mm = 1;
      yy  = 1;
    }
    return yy.toString()   "-"   mm.toString().padStart(2, "0");
  }
  if (result.length > 0) {
    for (let yymm = addmonth(result[result.length - 1].Month); yymm < item.Month; yymm = addmonth(yymm)) {
      result.push({
        Month: yymm,
        Count: 0
      });
    }
  }
  result.push(item);
});
console.log(result);

  • Related