Moment: El cálculo de la edad mínima con valores de cumpleaños límite da como resultado un comportamiento inesperado

Creado en 27 feb. 2017  ·  3Comentarios  ·  Fuente: moment/moment

Descripción del problema y pasos para reproducirlo

Quiero comprobar si una persona es exactamente mayor de 18 años. Entonces usé las funciones diff() y duration() para calcular la edad de la persona. Escribí algunas pruebas con valores límite para probar mi lógica. Quería probar si soy exactamente un día menor de 18 y exactamente 18. Así que elegí dos cumpleaños para la prueba (hoy es 27.02.2017):

  • exactamente 18: 27.02.1999
  • un día menor de 18 años: 28.02.1999

Reproducir

// today date is: 27.02.2017

// exactly age 18    
var birthday = moment("27.02.1999", "DD.MM.YYYY"),
    age = moment.duration(moment().diff(birthday)).asYears();

      console.log(age); // output: 18.00325100985672; expected: == 18

// one day under age 18
var birthday = moment("28.02.1999", "DD.MM.YYYY"),
    age = moment.duration(moment().diff(birthday)).asYears();

    console.log(age); // output: 18.00053005036735; expected: < 18

Creo que el cálculo es correcto, debido a los años bisiestos. Pero en términos de cumpleaños, la persona no debe tener 18 años. Tal vez haya otra forma de hacerlo con momentjs y lo estoy haciendo mal. Si es así, sería bueno ser mencionado en la documentación de momentjs.

Codigo de producción

<form id="newsletter-form" data-parsley-validate="">
  <input id="birthday" type="text" class="form-control" required="" data-parsley-minage="18">
  <input type="submit" class="btn btn-default" value="subscribe">
</form>

<script type="text/javascript">
    $(function () {
        window.Parsley.addValidator('minage', {
            validateString: function(value, minAge) {
                var birthday = moment(value, "DD.MM.YYYY"),
                    age = moment.duration(moment().diff(birthday)).asYears();

                return (age >= minAge);
            },
            requirementType: 'integer',
            messages: {
                en: 'You must be mature.'
            }
        });
    });
</script>
<script type="text/javascript">
    $(function () {
        $('#newsletter-form').parsley()
            // field validation
            .on('form:submit', function() {
                return false; // do not submit
            });
    });
</script>

Medio ambiente

Versión de Chrome 56.0.2924.87 (64 bits) en Windows 7

Otra información que puede ser útil

  • Zona horaria de la máquina: "(UTC + 01: 00) Ámsterdam, Berlín, Berna, Rom, Estocolmo, Viena"
  • La hora y la fecha en que se ejecutó el código: "Lunes 27 de febrero de 2017 13:38:04 GMT + 0100"
  • Otras bibliotecas en uso: jquery-3.1.1, bootstrap-3.3.7, bootstrap-datepicker-1.6.4, parselyjs-2.6.2
console.log( (new Date()).toString()) // Mon Feb 27 2017 13:38:04 GMT+0100 (Mitteleuropäische Zeit)
console.log((new Date()).toLocaleString()) // 27.2.2017, 13:38:04
console.log( (new Date()).getTimezoneOffset()) // -60
console.log( navigator.userAgent) // Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/56.0.2924.87 Safari/537.36
console.log(moment.version) // 2.17.1

Comentario más útil

¿Funciona mejor si solo usa

age = moment().diff(birthday, 'years');

Todos 3 comentarios

Creo que el cálculo es incorrecto, pero está muy cerca. ¿Quizás debido a errores de redondeo?

Por ejemplo:

a = moment("27.02.1999", "DD.MM.YYYY");
b = moment("28.02.1999", "DD.MM.YYYY");
c = moment("27.02.2017", "DD.MM.YYYY");
moment.duration(c.diff(a)).asYears(); // 18.00173857094944
moment.duration(c.diff(b)).asYears(); // 17.99900066394245

Entonces, en este caso, moment.duration(end.diff(start)).asYears() devolverá 18 alrededor de 16 horas antes de lo debido.

Si está tratando de verificar si una edad es mayor o igual a 18, puede comparar los años, luego (si hay 18 años de diferencia) los meses, luego los días.

// pseudocode, correctness not guaranteed
function is18(start, end) {
  if (end.year() - start.year() == 18) {
    if (end.month() == start.month()) {
      return end.date() >= start.date();
    }
    return end.month() > start.month();
  }
  return end.year() - 18 > start.year();
}

¿Funciona mejor si solo usa

age = moment().diff(birthday, 'years');

@ mj1856

Sí, esto funciona bien.

// today date is: 03.03.2017

// exactly age 18    
var birthday = moment("03.03.1999", "DD.MM.YYYY"),
    age = moment().diff(birthday, 'years');

      console.log(age); // output: 18; expected: == 18

// one day under age 18
var birthday = moment("04.03.1999", "DD.MM.YYYY"),
    age = moment().diff(birthday, 'years');

    console.log(age); // output: 17; expected: < 18
¿Fue útil esta página
0 / 5 - 0 calificaciones