Поскольку определение "возраст в месяцах"... гибкое, самый простой способ-использовать небольшую арифметику, как вы бы вычислили ее в уме, и не включать Date
класс.
Для [а] человеческой интерпретации "возраста в месяцах" правило таково
Вычислите разницу между двумя датами в месяцах,
как если бы день месяца был 1-м числом месяца для обеих дат
Вычтите 1, чтобы исключить последний месяц
Затем, если день месяца последнего дня периода приходится на
день месяца первого дня периода или после него, [потенциально частичный] последний месяц завершен: добавьте 1, чтобы восстановить счетчик
Одна ложка дегтя в бочке меда заключается в том, что месяцы содержат разное количество дней, что касается случаев, когда 2 месяца отличаются по количеству дней.
Однако, если конечный месяц короче начального, вы можете оказаться в ситуации, когда граничное условие никогда не может быть выполнено (например, дата начала-28 февраля, а дата окончания-31 марта. Чтобы исправить это, вам нужно посмотреть на "конец месяца" как на окно в диапазоне от последнего дня начального месяца до последнего дня последнего месяца включительно.
Это приводит к следующему коду. Я использую структуру, подобную следующей, для представления даты:
{
year: 2021 , // 4-digit year
month: 11 , // month of year (1-12 mapping to January-December)
day: 23 // day of month (1-[28-31] depending on year/month
}
Обеспечение того, чтобы данные в этой структуре представляли действительную дату, оставляется в качестве упражнения для читателя.
Код не такой уж сложный:
/**
*
* @param {object} bgn - start date of period
* @param {number} bgn.year - 4-digit year
* @param {number} bgn.month - month of year [1-12]
* @param {number} bgn.day - day of month [1-31]
*
* @param {object} end - end date of period
* @param {number} end.year - 4-digit year
* @param {number} end.month - month of year [1-12]
* @param {number} end.day - day of month [1-31]
*
*/
function diffInMonths( bgn , end ) {
const between = ( x , min , max ) => x >= min && x <= max;
// We'll need to add back the final month based on the following:
// - end.day >= bgn.day -- we've passed the month boundary, or
// - end.day is within the end-of-month window
// (when the end month is shorter than the start month)
const needAdjustment = end.day >= bgn.day
|| between( end.day, daysInMonth(bgn), daysInMonth(end) );
const finalMonthAdjustment = needsAdjustment ? 1 : 0;
const deltaM = 12 * ( end.year - bgn.year )
+ ( end.month - bgn.month )
- 1 // remove the final month from the equation
+ finalMonthAdjustment // add in the precomputed final month adjustment
;
return deltaM;
}
/**
*
* @param {object} dt - date
* @param {number} dt.year - 4-digit year
* @param {number} dt.month - month of year [1-12]
* @param {number} dt.day - day of month [1-31]
*
*/
function daysInMonth(dt) {
const leapYear = ( dt.year % 4 === 0 && dt.year % 100 !== 0 ) || dt.year % 400 === 0;
const monthDays = leapYear ? daysPerMonthLeap : daysPerMonth;
const days = monthDays[dt.month];
return days;
}
// jan feb mar apr may jun jul aug sep oct nov dec
// ---------- --- --- --- --- --- --- --- --- --- --- --- ---
const daysPerMonth = [ undefined, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31, ];
const daysPerMonthLeap = [ undefined, 31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31, ];
new Date(userDate.split('/').reverse())
почти наверняка вернет неверную дату.