¿Cómo calculo la edad de alguien en función de un cumpleaños tipo DateTime?

Dado un DateTime representando el cumpleaños de una persona, ¿cómo calculo su edad en años?

preguntado el 31 de julio de 08 a las 21:07

lo que se han perdido todas las respuestas hasta ahora es que depende de dónde nació la persona y dónde se encuentra en este momento. -

@Yaur: Simplemente convierta la hora de ahora + nacimiento a GMT / UTC, la edad es solo un valor relativo, por lo tanto, las zonas horarias son irrelevantes. Para determinar la zona horaria actual del usuario, puede utilizar GeoLocating. -

Si estamos tomando en consideración la sugerencia de @Yaur de los cálculos entre zonas horarias, ¿debería el horario de verano afectar el cálculo de alguna manera? -

Tenga en cuenta que para una persona menor de un año, su edad se expresa en días, semanas o meses. El tiempo de transición de las unidades puede ser específico del dominio. -

Como todos podemos ver, no existe una definición definitiva de edad. Muchas mujeres que he conocido tienden a redondear su tiempo de vida a un año completo hasta los veintitantos, luego comienzan a redondear hacia abajo. Nací el 3 de enero, así que solo resto el año actual de mi año de nacimiento, sin importar el día que sea. Algunas personas piensan que si naciste en un día bisiesto, envejeces en una proporción de 1/4. ¿Y si nacieras en un segundo bisiesto? ¿Un bebé de 8 meses cuenta como 1? Si vuelo hacia el oeste, ¿me hago más joven? Si mi corazón se detiene por un minuto, ¿debería incluir eso en el cálculo? -

30 Respuestas

Una solución sencilla y fácil de entender.

// Save today's date.
var today = DateTime.Today;

// Calculate the age.
var age = today.Year - birthdate.Year;

// Go back to the year in which the person was born in case of a leap year
if (birthdate.Date > today.AddYears(-age)) age--;

Sin embargo, esto supone que está buscando el occidental idea de la edad y no usar Ajuste de cuentas de Asia oriental.

Respondido el 07 de Septiembre de 20 a las 10:09

Esta respuesta no funciona con todas las configuraciones regionales y todas las edades. Varios países se han saltado fechas después del nacimiento de las personas vivas actuales, incluidos Rusia (1918), Grecia (1924) y Turquía (1926). - lars d

De hecho, todavía no es del todo correcto. Este código supone que 'bday' es la parte de fecha de un DateTime. Es un caso límite (supongo que la mayoría de las personas solo pasarán fechas y no fechas y horas), pero si pasa un cumpleaños como una fecha y hora donde la hora es mayor que 00:00:00, entonces ' Me toparé con el error que señaló Danvil. Establecer bday = bday.Date soluciona esto. - Øyvind

@NKCSS: Eso no maneja correctamente el año bisiesto. Mis pruebas muestran que solo if (bday > now.AddYears(-age)) obras. Desde DayOfYear devoluciones 61 en 2012 y 60 en 2011 para el 1 de marzo. - Guvante

¿No sería más fácil simplemente hacer DateTime dateDifference = subject.Birthday - DateTime.Now; int age = dateDifference.Year;? Además, no creo que la edad dependa de los años bisiestos. Después de todo, un año se define como 365 Nolonar

En Inglaterra y Gales o en Hong Kong, una persona nacida el 29 de febrero habrá cumplido legalmente los 18 años el 1 de marzo. Si nació en Taiwán, cumplirá legalmente 18 años el 28 de febrero, un día antes. (Wikipedia ) Entonces birthdate.AddYears(age) > today y birthdate > today.AddYears(-age) son correctos e incorrectos, según la jurisdicción. - daniel luzzi

Esta es una forma extraña de hacerlo, pero si formatea la fecha para yyyymmdd y reste la fecha de nacimiento de la fecha actual y luego suelte los últimos 4 dígitos que tiene la edad :)

No sé C #, pero creo que esto funcionará en cualquier idioma.

20080814 - 19800703 = 280111 

Suelta los últimos 4 dígitos = 28.

Código C #:

int now = int.Parse(DateTime.Now.ToString("yyyyMMdd"));
int dob = int.Parse(dateOfBirth.ToString("yyyyMMdd"));
int age = (now - dob) / 10000;

O alternativamente sin toda la conversión de tipos en forma de método de extensión. Comprobación de errores omitida:

public static Int32 GetAge(this DateTime dateOfBirth)
{
    var today = DateTime.Today;

    var a = (today.Year * 100 + today.Month) * 100 + today.Day;
    var b = (dateOfBirth.Year * 100 + dateOfBirth.Month) * 100 + dateOfBirth.Day;

    return (a - b) / 10000;
}

Respondido el 24 de diciembre de 19 a las 08:12

En realidad, esto es excelente para su uso en MS-SQL con campos de fecha y hora (días totales desde 01-011900) - Patrik

en su respuesta alternativa, puede evitar el desbordamiento de enteros restando los años, luego restando el mes * 30.5 + día y dividiendo por 366 - numerek

@numerek Publique sus modificaciones sugeridas como su propia respuesta. Por lo que vale, el año actual multiplicado por 10000 no se acerca a un desbordamiento de enteros, en dos órdenes de magnitud. 20,150,000 vs 2,147,483,648 - GalácticoVaquero

Sería mejor agregar un parámetro: float.Parse (x, CultureInfo.InvariantCulture) porque algunos lugareños usan una coma en lugar de un punto para un delimitador en números fraccionarios - Vlad Krilov

@LongChalk 20180101 - 20171231 = 8870. Suelta los últimos 4 dígitos y tendrás (un implícito) 0 para la edad. Como llegaste 1? - rufo l

Aquí hay un fragmento de prueba:

DateTime bDay = new DateTime(2000, 2, 29);
DateTime now = new DateTime(2009, 2, 28);
MessageBox.Show(string.Format("Test {0} {1} {2}",
                CalculateAgeWrong1(bDay, now),      // outputs 9
                CalculateAgeWrong2(bDay, now),      // outputs 9
                CalculateAgeCorrect(bDay, now),     // outputs 8
                CalculateAgeCorrect2(bDay, now)));  // outputs 8

Aquí tienes los métodos:

public int CalculateAgeWrong1(DateTime birthDate, DateTime now)
{
    return new DateTime(now.Subtract(birthDate).Ticks).Year - 1;
}

public int CalculateAgeWrong2(DateTime birthDate, DateTime now)
{
    int age = now.Year - birthDate.Year;

    if (now < birthDate.AddYears(age))
        age--;

    return age;
}

public int CalculateAgeCorrect(DateTime birthDate, DateTime now)
{
    int age = now.Year - birthDate.Year;

    if (now.Month < birthDate.Month || (now.Month == birthDate.Month && now.Day < birthDate.Day))
        age--;

    return age;
}

public int CalculateAgeCorrect2(DateTime birthDate, DateTime now)
{
    int age = now.Year - birthDate.Year;

    // For leap years we need this
    if (birthDate > now.AddYears(-age)) 
        age--;
    // Don't use:
    // if (birthDate.AddYears(age) > now) 
    //     age--;

    return age;
}

Respondido 29 Feb 20, 01:02

Si bien este código funciona, afirma que una persona nacida en un día bisiesto alcanza el próximo año de edad el 1 de marzo en los años no bisiestos, en lugar del 28 de febrero. En realidad, cualquiera de las opciones puede ser correcta. Wikipedia tiene algo que decir al respecto.. Entonces, si bien su código no es "incorrecto", tampoco lo es la solución aceptada. - Matt Johnson-Pinta

@MattJohnson Creo que eso es correcto. Si mi cumpleaños fue el 29 de febrero, entonces el 28 de febrero no ha pasado, y todavía debería tener la misma edad que el 27 de febrero. El 1 de marzo, sin embargo, hemos pasado mi cumpleaños y debería tener la siguiente edad. En los EE. UU., Un negocio que vende alcohol tendrá un letrero que diga algo como "Si nació después de este día en YYYY, no puede comprar alcohol" (donde YYYY cambia cada año). Eso significa que una persona nacida el 29 de febrero no puede comprar alcohol el 28 de febrero del año en que cumple 21 años (en la mayoría de los lugares), y apoya la idea de que no es un año mayor hasta el 1 de marzo. jfren484

@ jfren484 - lea el artículo de Wikipedia. Varía considerablemente entre jurisdicciones. - Matt Johnson-Pinta

@ jfren484 Su afirmación no tiene absolutamente nada que ver con la filosofía; pero todo lo que tiene que ver con tu propio sentimiento personal. Cuando una persona nacida el 29 de febrero, las "edades" carecen en gran medida de importancia a menos que la edad forme un "límite de edad legal" (por ejemplo, puede comprar alcohol, votar, obtener una pensión, unirse al ejército, obtener una licencia de conducir). Considere la edad para beber en los EE. UU. (21 años): para la mayoría de las personas eso es 7670 días. Son 7671 días si nació antes del 29 de febrero en un año bisiesto o desde el 1 de marzo antes del año bisiesto. Si nació el 29 de febrero: 28 de febrero son 7670 días y 1 de marzo son 7671 días. La elección es arbitraria puede ir de cualquier manera. - Desilusionado

@CraigYoung No entiendes lo que quise decir filosóficamente. Usé ese término en contraste con legal. Si uno está escribiendo una solicitud que necesita saber la edad legal de una persona, entonces todo lo que necesita saber es cómo las jurisdicciones legales en las que se usa su solicitud / para tratar a las personas nacidas el 29 de febrero. Sin embargo, si estamos hablando de como eso debemos ser tratado, entonces eso es por definición, filosofía. Y sí, la opinión que di es mi propia opinión, pero como dije, creo que sería más fácil defender el 1 de marzo que el 28 de febrero. jfren484

La respuesta simple a esto es aplicar AddYears como se muestra a continuación porque este es el único método nativo para agregar años al 29 de febrero de los años bisiestos y obtener el resultado correcto del 28 de febrero para los años comunes.

Algunos sienten que el 1 de marzo es el cumpleaños de los saltos, pero ni .Net ni ninguna regla oficial lo respalda, ni la lógica común explica por qué algunos nacidos en febrero deberían tener el 75% de sus cumpleaños en otro mes.

Además, un método Age se presta a ser agregado como una extensión a DateTime. De esta forma puedes obtener la edad de la forma más sencilla posible:

  1. Artículo de lista

int edad = fecha de nacimiento.Edad ();

public static class DateTimeExtensions
{
    /// <summary>
    /// Calculates the age in years of the current System.DateTime object today.
    /// </summary>
    /// <param name="birthDate">The date of birth</param>
    /// <returns>Age in years today. 0 is returned for a future date of birth.</returns>
    public static int Age(this DateTime birthDate)
    {
        return Age(birthDate, DateTime.Today);
    }

    /// <summary>
    /// Calculates the age in years of the current System.DateTime object on a later date.
    /// </summary>
    /// <param name="birthDate">The date of birth</param>
    /// <param name="laterDate">The date on which to calculate the age.</param>
    /// <returns>Age in years on a later day. 0 is returned as minimum.</returns>
    public static int Age(this DateTime birthDate, DateTime laterDate)
    {
        int age;
        age = laterDate.Year - birthDate.Year;

        if (age > 0)
        {
            age -= Convert.ToInt32(laterDate.Date < birthDate.Date.AddYears(age));
        }
        else
        {
            age = 0;
        }

        return age;
    }
}

Ahora, ejecute esta prueba:

class Program
{
    static void Main(string[] args)
    {
        RunTest();
    }

    private static void RunTest()
    {
        DateTime birthDate = new DateTime(2000, 2, 28);
        DateTime laterDate = new DateTime(2011, 2, 27);
        string iso = "yyyy-MM-dd";

        for (int i = 0; i < 3; i++)
        {
            for (int j = 0; j < 3; j++)
            {
                Console.WriteLine("Birth date: " + birthDate.AddDays(i).ToString(iso) + "  Later date: " + laterDate.AddDays(j).ToString(iso) + "  Age: " + birthDate.AddDays(i).Age(laterDate.AddDays(j)).ToString());
            }
        }

        Console.ReadKey();
    }
}

El ejemplo de fecha crítica es este:

Fecha de nacimiento: 2000-02-29 Fecha posterior: 2011-02-28 Edad: 11

Salida:

{
    Birth date: 2000-02-28  Later date: 2011-02-27  Age: 10
    Birth date: 2000-02-28  Later date: 2011-02-28  Age: 11
    Birth date: 2000-02-28  Later date: 2011-03-01  Age: 11
    Birth date: 2000-02-29  Later date: 2011-02-27  Age: 10
    Birth date: 2000-02-29  Later date: 2011-02-28  Age: 11
    Birth date: 2000-02-29  Later date: 2011-03-01  Age: 11
    Birth date: 2000-03-01  Later date: 2011-02-27  Age: 10
    Birth date: 2000-03-01  Later date: 2011-02-28  Age: 10
    Birth date: 2000-03-01  Later date: 2011-03-01  Age: 11
}

Y para la fecha posterior 2012-02-28:

{
    Birth date: 2000-02-28  Later date: 2012-02-28  Age: 12
    Birth date: 2000-02-28  Later date: 2012-02-29  Age: 12
    Birth date: 2000-02-28  Later date: 2012-03-01  Age: 12
    Birth date: 2000-02-29  Later date: 2012-02-28  Age: 11
    Birth date: 2000-02-29  Later date: 2012-02-29  Age: 12
    Birth date: 2000-02-29  Later date: 2012-03-01  Age: 12
    Birth date: 2000-03-01  Later date: 2012-02-28  Age: 11
    Birth date: 2000-03-01  Later date: 2012-02-29  Age: 11
    Birth date: 2000-03-01  Later date: 2012-03-01  Age: 12
}

Respondido 07 Feb 16, 00:02

Un comentario sobre tener el cumpleaños del 29 de febrero el 1 de marzo, técnicamente, tenerlo el 28 es demasiado pronto (de hecho, 1 día antes). El día 1 es un día demasiado tarde. Pero dado que el cumpleaños está entre, usar el 1er para calcular la edad en años no bisiestos tiene más sentido para mí, ya que esa persona tiene esa edad el 1 de marzo (y 2 y 3) de cada año, pero no el 28 de febrero. - CyberClaw

Desde el punto de vista del diseño de software, escribir esto como un método de extensión no tiene mucho sentido para mí. date.Age(other)? - Marsze

Mi sugerencia

int age = (int) ((DateTime.Now - bday).TotalDays/365.242199);

Eso parece tener el año cambiando en la fecha correcta. (Hice pruebas puntuales hasta los 107 años).

Respondido 29 Feb 20, 01:02

No creo que Harry Patch hubiera apreciado su metodología de prueba puntual: latimes.com/news/obituaries/… - MusiGénesis

Google dice days in a year = 365.242199 - mpen

La duración media de un año en el calendario gregoriano es de 365.2425 días. - dan04

Yo diría que esta es una de las soluciones más simples y es lo suficientemente bueno. A quién le importa si estoy medio día antes de mi X cumpleaños y el programa dice que tengo X años. El programa es más o menos acertado, aunque no matemáticamente. Me gusta mucho esta solución. - Peter Perhač

^^ Porque a veces es importante. En mis pruebas, esto falla en el cumpleaños de la persona, los reporta más jóvenes de lo que son. - ChadT

Otra función, no por mí, pero la encontré en la web y la refiné un poco:

public static int GetAge(DateTime birthDate)
{
    DateTime n = DateTime.Now; // To avoid a race condition around midnight
    int age = n.Year - birthDate.Year;

    if (n.Month < birthDate.Month || (n.Month == birthDate.Month && n.Day < birthDate.Day))
        age--;

    return age;
}

Solo dos cosas que me vienen a la mente: ¿Qué pasa con las personas de países que no usan el calendario gregoriano? DateTime.Now está en la cultura específica del servidor, creo. No tengo absolutamente ningún conocimiento sobre cómo trabajar con calendarios asiáticos y no sé si hay una manera fácil de convertir fechas entre calendarios, pero por si acaso te estás preguntando acerca de esos chicos chinos del año 4660 :-)

Respondido 29 Feb 20, 01:02

Esto parece manejar mejor diferentes regiones (formatos de fecha). - webdad3

2 Los principales problemas a resolver son:

1. Calcule la edad exacta - en años, meses, días, etc.

2. Calcular la edad percibida generalmente - A las personas generalmente no les importa la edad que tengan exactamente, solo les importa cuándo es su cumpleaños en el año en curso.


Solución para 1 es obvio:

DateTime birth = DateTime.Parse("1.1.2000");
DateTime today = DateTime.Today;     //we usually don't care about birth time
TimeSpan age = today - birth;        //.NET FCL should guarantee this as precise
double ageInDays = age.TotalDays;    //total number of days ... also precise
double daysInYear = 365.2425;        //statistical value for 400 years
double ageInYears = ageInDays / daysInYear;  //can be shifted ... not so precise

Solución para 2 es la que no es tan precisa para determinar la edad total, pero la gente la percibe como precisa. La gente también suele utilizarlo cuando calculan su edad "manualmente":

DateTime birth = DateTime.Parse("1.1.2000");
DateTime today = DateTime.Today;
int age = today.Year - birth.Year;    //people perceive their age in years

if (today.Month < birth.Month ||
   ((today.Month == birth.Month) && (today.Day < birth.Day)))
{
  age--;  //birthday in current year not yet reached, we are 1 year younger ;)
          //+ no birthday for 29.2. guys ... sorry, just wrong date for birth
}

Notas a 2 .:

  • Esta es mi solución preferida
  • No podemos usar DateTime.DayOfYear o TimeSpans, ya que cambian el número de días en años bisiestos
  • He puesto un poco más de líneas para facilitar la lectura.

Solo una nota más ... crearía 2 métodos sobrecargados estáticos para él, uno para uso universal, el segundo para facilidad de uso:

public static int GetAge(DateTime bithDay, DateTime today) 
{ 
  //chosen solution method body
}

public static int GetAge(DateTime birthDay) 
{ 
  return GetAge(birthDay, DateTime.Now);
}

Respondido 07 Feb 16, 00:02

Aquí hay una sola línea:

int age = new DateTime(DateTime.Now.Subtract(birthday).Ticks).Year-1;

Respondido 29 Feb 20, 01:02

Esto está descompuesto. Hecho comprobable: public static int CalculateAge (DateTime dateOfBirth, DateTime dateToCalculateAge) {return new DateTime (dateToCalculateAge.Subtract (dateOfBirth) .Ticks) .Year - 1; } ... Da 14 años cuando ingreso 1990-06-01 y calculo la edad el día ANTES de que cumpla 14 años (1990-05-31). - Kjensen

La mejor forma que conozco debido a los años bisiestos y todo es:

DateTime birthDate = new DateTime(2000,3,1);
int age = (int)Math.Floor((DateTime.Now - birthDate).TotalDays / 365.25D);

Respondido 29 Feb 20, 01:02

Esta es la versión que usamos aquí. Funciona y es bastante simple. Es la misma idea que la de Jeff, pero creo que es un poco más clara porque separa la lógica para restar uno, por lo que es un poco más fácil de entender.

public static int GetAge(this DateTime dateOfBirth, DateTime dateAsAt)
{
    return dateAsAt.Year - dateOfBirth.Year - (dateOfBirth.DayOfYear < dateAsAt.DayOfYear ? 0 : 1);
}

Puede expandir el operador ternario para hacerlo aún más claro, si cree que ese tipo de cosas no están claras.

Obviamente, esto se hace como un método de extensión en DateTime, pero claramente puede tomar esa línea de código que hace el trabajo y ponerla en cualquier lugar. Aquí tenemos otra sobrecarga del método Extension que pasa DateTime.Now, solo para completar.

Respondido el 24 de diciembre de 19 a las 08:12

Creo que esto puede fallar un día cuando exactamente uno de dateOfBirth o dateAsAt cae en un año bisiesto. Considere la edad de una persona nacida el 1 de marzo de 2003 el 29 de febrero de 2004. Para rectificar esto, necesita hacer una comparación lexicográfica de pares (Month, DayOfMonth) y usarla para el condicional. - Doug McClean

tampoco mostrará la edad correcta en el momento de su cumpleaños. - dotjoe

Esto le da "más detalles" a esta pregunta. Quizás esto es lo que estás buscando

DateTime birth = new DateTime(1974, 8, 29);
DateTime today = DateTime.Now;
TimeSpan span = today - birth;
DateTime age = DateTime.MinValue + span;

// Make adjustment due to MinValue equalling 1/1/1
int years = age.Year - 1;
int months = age.Month - 1;
int days = age.Day - 1;

// Print out not only how many years old they are but give months and days as well
Console.Write("{0} years, {1} months, {2} days", years, months, days);

Respondido 15 Oct 13, 13:10

Esto no funciona todo el tiempo. Agregar un intervalo a DateTime.MinValue podría funcionar boes, esto no tiene en cuenta los años bisiestos, etc. Si agrega los años, meses y días a Age usando la función AddYears (), AddMonths y AddDays (), no siempre devolverá la fecha y hora .Ahora fecha. - Atanasios Kataras

El intervalo de tiempo en sí mismo tiene en cuenta automáticamente los años bisiestos entre 2 fechas, por lo que no estoy seguro de qué está hablando. He preguntado en foros de microsoft y microsoft ha confirmado que tiene en cuenta los años bisiestos entre 2 fechas. - Jacqueline Loriault

Considere los siguientes DOS escenarios. 1st DateTime. Ahora es 1/1/2001 y un niño nace el 1/1/2000. 2000 es un año bisiesto y el resultado será 1 año, 0 meses y 1 día. En el segundo período, DateTime. Ahora es 1/1/2002 y el niño nace el 1/1/2001. En este caso, el resultado será 1 año, 0 meses y 0 días. Eso sucederá porque está agregando el período de tiempo en un año no bisiesto. Si DateTime.MinValue fuera un año bisiesto, los resultados serían 1 año en el primer año y 0 años 11 meses y 30 días. (Pruébelo en su código). - Atanasios Kataras

¡Vota a favor! Se me ocurrió una solución que es prácticamente idéntica (utilicé DateTime.MinValue.AddTicks (span.Ticks) en lugar de +, pero el resultado es el mismo y el tuyo tiene algunos caracteres menos código). - makotosan

Tienes toda la razón, no lo es. Pero si lo fuera, ese sería el resultado. ¿Por qué eso importa? No es así. En cualquier caso, salte o no, hay ejemplos en los que esto no funciona. Eso era lo que quería mostrar. El DIFF es correcto. Span tiene en cuenta los años bisiestos. Pero AGREGAR a una fecha base no lo es. Pruebe los ejemplos en código y verá que tengo razón. - Atanasios Kataras

Yo uso esto:

public static class DateTimeExtensions
{
    public static int Age(this DateTime birthDate)
    {
        return Age(birthDate, DateTime.Now);
    }

    public static int Age(this DateTime birthDate, DateTime offsetDate)
    {
        int result=0;
        result = offsetDate.Year - birthDate.Year;

        if (offsetDate.DayOfYear < birthDate.DayOfYear)
        {
              result--;
        }

        return result;
    }
}

Respondido el 03 de junio de 15 a las 12:06

He creado una función definida por el usuario de SQL Server para calcular la edad de alguien, dada su fecha de nacimiento. Esto es útil cuando lo necesita como parte de una consulta:

using System;
using System.Data;
using System.Data.Sql;
using System.Data.SqlClient;
using System.Data.SqlTypes;
using Microsoft.SqlServer.Server;

public partial class UserDefinedFunctions
{
    [SqlFunction(DataAccess = DataAccessKind.Read)]
    public static SqlInt32 CalculateAge(string strBirthDate)
    {
        DateTime dtBirthDate = new DateTime();
        dtBirthDate = Convert.ToDateTime(strBirthDate);
        DateTime dtToday = DateTime.Now;

        // get the difference in years
        int years = dtToday.Year - dtBirthDate.Year;

        // subtract another year if we're before the
        // birth day in the current year
        if (dtToday.Month < dtBirthDate.Month || (dtToday.Month == dtBirthDate.Month && dtToday.Day < dtBirthDate.Day))
            years=years-1;

        int intCustomerAge = years;
        return intCustomerAge;
    }
};

Respondido 07 Feb 16, 00:02

Aquí hay otra respuesta más:

public static int AgeInYears(DateTime birthday, DateTime today)
{
    return ((today.Year - birthday.Year) * 372 + (today.Month - birthday.Month) * 31 + (today.Day - birthday.Day)) / 372;
}

Esto ha sido probado exhaustivamente por unidades. Parece un poco "mágico". El número 372 es el número de días que habría en un año si cada mes tuviera 31 días.

La explicación de por qué funciona (levantado de aquí) es:

Vamos a poner Yn = DateTime.Now.Year, Yb = birthday.Year, Mn = DateTime.Now.Month, Mb = birthday.Month, Dn = DateTime.Now.Day, Db = birthday.Day

age = Yn - Yb + (31*(Mn - Mb) + (Dn - Db)) / 372

Sabemos que lo que necesitamos es Yn-Yb si ya se ha alcanzado la fecha, Yn-Yb-1 si no es así.

a) Si Mn<Mb, Tenemos -341 <= 31*(Mn-Mb) <= -31 and -30 <= Dn-Db <= 30

-371 <= 31*(Mn - Mb) + (Dn - Db) <= -1

Con división de enteros

(31*(Mn - Mb) + (Dn - Db)) / 372 = -1

b) Si Mn=Mb y Dn<Db, Tenemos 31*(Mn - Mb) = 0 and -30 <= Dn-Db <= -1

Con división de enteros, de nuevo

(31*(Mn - Mb) + (Dn - Db)) / 372 = -1

c) Si Mn>Mb, Tenemos 31 <= 31*(Mn-Mb) <= 341 and -30 <= Dn-Db <= 30

1 <= 31*(Mn - Mb) + (Dn - Db) <= 371

Con división de enteros

(31*(Mn - Mb) + (Dn - Db)) / 372 = 0

d) Si Mn=Mb y Dn>Db, Tenemos 31*(Mn - Mb) = 0 and 1 <= Dn-Db <= 30

Con división de enteros, de nuevo

(31*(Mn - Mb) + (Dn - Db)) / 372 = 0

e) Si Mn=Mb y Dn=Db, Tenemos 31*(Mn - Mb) + Dn-Db = 0

y por lo tanto (31*(Mn - Mb) + (Dn - Db)) / 372 = 0

Respondido el 20 de junio de 20 a las 10:06

Pasé un tiempo trabajando en esto y se me ocurrió esto para calcular la edad de alguien en años, meses y días. Probé el problema del 29 de febrero y los años bisiestos y parece que funciona, agradecería cualquier comentario:

public void LoopAge(DateTime myDOB, DateTime FutureDate)
{
    int years = 0;
    int months = 0;
    int days = 0;

    DateTime tmpMyDOB = new DateTime(myDOB.Year, myDOB.Month, 1);

    DateTime tmpFutureDate = new DateTime(FutureDate.Year, FutureDate.Month, 1);

    while (tmpMyDOB.AddYears(years).AddMonths(months) < tmpFutureDate)
    {
        months++;

        if (months > 12)
        {
            years++;
            months = months - 12;
        }
    }

    if (FutureDate.Day >= myDOB.Day)
    {
        days = days + FutureDate.Day - myDOB.Day;
    }
    else
    {
        months--;

        if (months < 0)
        {
            years--;
            months = months + 12;
        }

        days +=
            DateTime.DaysInMonth(
                FutureDate.AddMonths(-1).Year, FutureDate.AddMonths(-1).Month
            ) + FutureDate.Day - myDOB.Day;

    }

    //add an extra day if the dob is a leap day
    if (DateTime.IsLeapYear(myDOB.Year) && myDOB.Month == 2 && myDOB.Day == 29)
    {
        //but only if the future date is less than 1st March
        if (FutureDate >= new DateTime(FutureDate.Year, 3, 1))
            days++;
    }

}

Respondido 07 Feb 16, 00:02

¿Necesitamos considerar a las personas menores de 1 año? Como cultura china, describimos la edad de los bebés pequeños como 2 meses o 4 semanas.

A continuación se muestra mi implementación, no es tan simple como lo imaginaba, especialmente para lidiar con una fecha como el 2/28.

public static string HowOld(DateTime birthday, DateTime now)
{
    if (now < birthday)
        throw new ArgumentOutOfRangeException("birthday must be less than now.");

    TimeSpan diff = now - birthday;
    int diffDays = (int)diff.TotalDays;

    if (diffDays > 7)//year, month and week
    {
        int age = now.Year - birthday.Year;

        if (birthday > now.AddYears(-age))
            age--;

        if (age > 0)
        {
            return age + (age > 1 ? " years" : " year");
        }
        else
        {// month and week
            DateTime d = birthday;
            int diffMonth = 1;

            while (d.AddMonths(diffMonth) <= now)
            {
                diffMonth++;
            }

            age = diffMonth-1;

            if (age == 1 && d.Day > now.Day)
                age--;

            if (age > 0)
            {
                return age + (age > 1 ? " months" : " month");
            }
            else
            {
                age = diffDays / 7;
                return age + (age > 1 ? " weeks" : " week");
            }
        }
    }
    else if (diffDays > 0)
    {
        int age = diffDays;
        return age + (age > 1 ? " days" : " day");
    }
    else
    {
        int age = diffDays;
        return "just born";
    }
}

Esta implementación ha pasado por debajo de los casos de prueba.

[TestMethod]
public void TestAge()
{
    string age = HowOld(new DateTime(2011, 1, 1), new DateTime(2012, 11, 30));
    Assert.AreEqual("1 year", age);

    age = HowOld(new DateTime(2011, 11, 30), new DateTime(2012, 11, 30));
    Assert.AreEqual("1 year", age);

    age = HowOld(new DateTime(2001, 1, 1), new DateTime(2012, 11, 30));
    Assert.AreEqual("11 years", age);

    age = HowOld(new DateTime(2012, 1, 1), new DateTime(2012, 11, 30));
    Assert.AreEqual("10 months", age);

    age = HowOld(new DateTime(2011, 12, 1), new DateTime(2012, 11, 30));
    Assert.AreEqual("11 months", age);

    age = HowOld(new DateTime(2012, 10, 1), new DateTime(2012, 11, 30));
    Assert.AreEqual("1 month", age);

    age = HowOld(new DateTime(2008, 2, 28), new DateTime(2009, 2, 28));
    Assert.AreEqual("1 year", age);

    age = HowOld(new DateTime(2008, 3, 28), new DateTime(2009, 2, 28));
    Assert.AreEqual("11 months", age);

    age = HowOld(new DateTime(2008, 3, 28), new DateTime(2009, 3, 28));
    Assert.AreEqual("1 year", age);

    age = HowOld(new DateTime(2009, 1, 28), new DateTime(2009, 2, 28));
    Assert.AreEqual("1 month", age);

    age = HowOld(new DateTime(2009, 2, 1), new DateTime(2009, 3, 1));
    Assert.AreEqual("1 month", age);

    // NOTE.
    // new DateTime(2008, 1, 31).AddMonths(1) == new DateTime(2009, 2, 28);
    // new DateTime(2008, 1, 28).AddMonths(1) == new DateTime(2009, 2, 28);
    age = HowOld(new DateTime(2009, 1, 31), new DateTime(2009, 2, 28));
    Assert.AreEqual("4 weeks", age);

    age = HowOld(new DateTime(2009, 2, 1), new DateTime(2009, 2, 28));
    Assert.AreEqual("3 weeks", age);

    age = HowOld(new DateTime(2009, 2, 1), new DateTime(2009, 3, 1));
    Assert.AreEqual("1 month", age);

    age = HowOld(new DateTime(2012, 11, 5), new DateTime(2012, 11, 30));
    Assert.AreEqual("3 weeks", age);

    age = HowOld(new DateTime(2012, 11, 1), new DateTime(2012, 11, 30));
    Assert.AreEqual("4 weeks", age);

    age = HowOld(new DateTime(2012, 11, 20), new DateTime(2012, 11, 30));
    Assert.AreEqual("1 week", age);

    age = HowOld(new DateTime(2012, 11, 25), new DateTime(2012, 11, 30));
    Assert.AreEqual("5 days", age);

    age = HowOld(new DateTime(2012, 11, 29), new DateTime(2012, 11, 30));
    Assert.AreEqual("1 day", age);

    age = HowOld(new DateTime(2012, 11, 30), new DateTime(2012, 11, 30));
    Assert.AreEqual("just born", age);

    age = HowOld(new DateTime(2000, 2, 29), new DateTime(2009, 2, 28));
    Assert.AreEqual("8 years", age);

    age = HowOld(new DateTime(2000, 2, 29), new DateTime(2009, 3, 1));
    Assert.AreEqual("9 years", age);

    Exception e = null;

    try
    {
        age = HowOld(new DateTime(2012, 12, 1), new DateTime(2012, 11, 30));
    }
    catch (ArgumentOutOfRangeException ex)
    {
        e = ex;
    }

    Assert.IsTrue(e != null);
}

Espero que sea de ayuda.

Respondido 07 Feb 16, 00:02

Esta no es una respuesta directa, sino más bien un razonamiento filosófico sobre el problema en cuestión desde un punto de vista cuasi científico.

Yo diría que la pregunta no especifica la unidad ni la cultura en la que medir la edad, la mayoría de las respuestas parecen asumir una representación anual entera. La unidad SI para el tiempo es second, ergo, la respuesta genérica correcta debería ser (por supuesto, suponiendo DateTime y sin tener en cuenta en absoluto los efectos relativistas):

var lifeInSeconds = (DateTime.Now.Ticks - then.Ticks)/TickFactor;

En la forma cristiana de calcular la edad en años:

var then = ... // Then, in this case the birthday
var now = DateTime.UtcNow;
int age = now.Year - then.Year;
if (now.AddYears(-age) < then) age--;

En finanzas, existe un problema similar cuando se calcula algo que a menudo se denomina Fracción de recuento de días, que es aproximadamente un número de años para un período determinado. Y la cuestión de la edad es realmente una cuestión de medir el tiempo.

Ejemplo de la convención real / real (contando todos los días "correctamente"):

DateTime start, end = .... // Whatever, assume start is before end

double startYearContribution = 1 - (double) start.DayOfYear / (double) (DateTime.IsLeapYear(start.Year) ? 366 : 365);
double endYearContribution = (double)end.DayOfYear / (double)(DateTime.IsLeapYear(end.Year) ? 366 : 365);
double middleContribution = (double) (end.Year - start.Year - 1);

double DCF = startYearContribution + endYearContribution + middleContribution;

Otra forma bastante común de medir el tiempo en general es "serializando" (el tipo que nombró esta convención de fechas debe haber estado tropezando seriamente):

DateTime start, end = .... // Whatever, assume start is before end
int days = (end - start).Days;

Me pregunto cuánto tiempo tenemos que pasar antes de que una era relativista en segundos se vuelva más útil que la aproximación aproximada de los ciclos de la Tierra alrededor del Sol durante la vida hasta ahora :) O en otras palabras, cuando un período debe tener una ubicación o una función que representa el movimiento por sí misma para ser válida :)

Respondido 31 Oct 17, 11:10

Qué es factor de marca? - Protiguo

@Protiguous Ticks por segundo, utilizado para normalizar los ticks a segundos. - Flindeberg

Manteniéndolo simple (y posiblemente estúpido :)).

DateTime birth = new DateTime(1975, 09, 27, 01, 00, 00, 00);
TimeSpan ts = DateTime.Now - birth;
Console.WriteLine("You are approximately " + ts.TotalSeconds.ToString() + " seconds old.");

Respondido el 24 de diciembre de 19 a las 08:12

TimeSpan fue mi primera opción, pero descubrí que no ofrece una propiedad TotalYears. Podría intentarlo (ts.TotalDays / 365), pero no tiene en cuenta los años bisiestos, etc. Lazlow

La forma más sencilla que he encontrado es esta. Funciona correctamente para las configuraciones regionales de EE. UU. Y Europa occidental. No puedo hablar con otros lugares, especialmente lugares como China. 4 comparaciones adicionales, como máximo, después del cálculo inicial de la edad.

public int AgeInYears(DateTime birthDate, DateTime referenceDate)
{
  Debug.Assert(referenceDate >= birthDate, 
               "birth date must be on or prior to the reference date");

  DateTime birth = birthDate.Date;
  DateTime reference = referenceDate.Date;
  int years = (reference.Year - birth.Year);

  //
  // an offset of -1 is applied if the birth date has 
  // not yet occurred in the current year.
  //
  if (reference.Month > birth.Month);
  else if (reference.Month < birth.Month) 
    --years;
  else // in birth month
  {
    if (reference.Day < birth.Day)
      --years;
  }

  return years ;
}

Estaba revisando las respuestas a esto y noté que nadie ha hecho referencia a las implicaciones regulatorias / legales de los nacimientos en días bisiestos. Por ejemplo, por Wikipedia, si nació el 29 de febrero en varias jurisdicciones, el cumpleaños de su año no bisiesto varía:

  • En el Reino Unido y Hong Kong: es el día ordinal del año, por lo que al día siguiente, el 1 de marzo es tu cumpleaños.
  • En Nueva Zelanda: es el día anterior, el 28 de febrero para los fines de la licencia de conducir y el 1 de marzo para otros fines.
  • Taiwán: es el 28 de febrero.

Y por lo que puedo decir, en los EE. UU., Los estatutos guardan silencio sobre el asunto, dejándolo en manos del derecho consuetudinario y de cómo varios organismos reguladores definen las cosas en sus regulaciones.

Para ello, una mejora:

public enum LeapDayRule
{
  OrdinalDay     = 1 ,
  LastDayOfMonth = 2 ,
}

static int ComputeAgeInYears(DateTime birth, DateTime reference, LeapYearBirthdayRule ruleInEffect)
{
  bool isLeapYearBirthday = CultureInfo.CurrentCulture.Calendar.IsLeapDay(birth.Year, birth.Month, birth.Day);
  DateTime cutoff;

  if (isLeapYearBirthday && !DateTime.IsLeapYear(reference.Year))
  {
    switch (ruleInEffect)
    {
      case LeapDayRule.OrdinalDay:
        cutoff = new DateTime(reference.Year, 1, 1)
                             .AddDays(birth.DayOfYear - 1);
        break;

      case LeapDayRule.LastDayOfMonth:
        cutoff = new DateTime(reference.Year, birth.Month, 1)
                             .AddMonths(1)
                             .AddDays(-1);
        break;

      default:
        throw new InvalidOperationException();
    }
  }
  else
  {
    cutoff = new DateTime(reference.Year, birth.Month, birth.Day);
  }

  int age = (reference.Year - birth.Year) + (reference >= cutoff ? 0 : -1);
  return age < 0 ? 0 : age;
}

Cabe señalar que este código asume:

  • Un cálculo de edad occidental (europeo), y
  • Un calendario, como el calendario gregoriano, que inserta un solo día bisiesto al final de un mes.

Respondido 05 Jul 18, 04:07

TimeSpan diff = DateTime.Now - birthdayDateTime;
string age = String.Format("{0:%y} years, {0:%M} months, {0:%d}, days old", diff);

No estoy seguro de cómo le gustaría que se le devolviera exactamente, así que hice una cadena legible.

Respondido el 24 de diciembre de 19 a las 08:12

He aquí una solución.

DateTime dateOfBirth = new DateTime(2000, 4, 18);
DateTime currentDate = DateTime.Now;

int ageInYears = 0;
int ageInMonths = 0;
int ageInDays = 0;

ageInDays = currentDate.Day - dateOfBirth.Day;
ageInMonths = currentDate.Month - dateOfBirth.Month;
ageInYears = currentDate.Year - dateOfBirth.Year;

if (ageInDays < 0)
{
    ageInDays += DateTime.DaysInMonth(currentDate.Year, currentDate.Month);
    ageInMonths = ageInMonths--;

    if (ageInMonths < 0)
    {
        ageInMonths += 12;
        ageInYears--;
    }
}

if (ageInMonths < 0)
{
    ageInMonths += 12;
    ageInYears--;
}

Console.WriteLine("{0}, {1}, {2}", ageInYears, ageInMonths, ageInDays);

Respondido 22 Feb 18, 16:02

Con string concat, esto sería posible: 47 años 11 meses 7 días - jose yates1980

Esta es una de las respuestas más precisas que puede resolver el cumpleaños del 29 de febrero en comparación con cualquier año del 28 de febrero.

public int GetAge(DateTime birthDate)
{
    int age = DateTime.Now.Year - birthDate.Year;

    if (birthDate.DayOfYear > DateTime.Now.DayOfYear)
        age--;

    return age;
}




Respondido 29 Feb 20, 01:02

¡Es hoy! (El próximo es dentro de cuatro años). Pedro Mortensen

Puede usar DateTime.Today en su lugar, ya que el tiempo no importa para el cálculo - tif

Tengo un método personalizado para calcular la edad, además de un mensaje de validación adicional en caso de que ayude:

public void GetAge(DateTime dob, DateTime now, out int years, out int months, out int days)
{
    years = 0;
    months = 0;
    days = 0;

    DateTime tmpdob = new DateTime(dob.Year, dob.Month, 1);
    DateTime tmpnow = new DateTime(now.Year, now.Month, 1);

    while (tmpdob.AddYears(years).AddMonths(months) < tmpnow)
    {
        months++;
        if (months > 12)
        {
            years++;
            months = months - 12;
        }
    }

    if (now.Day >= dob.Day)
        days = days + now.Day - dob.Day;
    else
    {
        months--;
        if (months < 0)
        {
            years--;
            months = months + 12;
        }
        days += DateTime.DaysInMonth(now.AddMonths(-1).Year, now.AddMonths(-1).Month) + now.Day - dob.Day;
    }

    if (DateTime.IsLeapYear(dob.Year) && dob.Month == 2 && dob.Day == 29 && now >= new DateTime(now.Year, 3, 1))
        days++;

}   

private string ValidateDate(DateTime dob) //This method will validate the date
{
    int Years = 0; int Months = 0; int Days = 0;

    GetAge(dob, DateTime.Now, out Years, out Months, out Days);

    if (Years < 18)
        message =  Years + " is too young. Please try again on your 18th birthday.";
    else if (Years >= 65)
        message = Years + " is too old. Date of Birth must not be 65 or older.";
    else
        return null; //Denotes validation passed
}

Llame al método aquí y pase el valor de fecha y hora (MM / dd / aaaa si el servidor está configurado en la configuración regional de EE.UU.). Reemplace esto con cualquier cosa que se muestre en un cuadro de mensaje o contenedor:

DateTime dob = DateTime.Parse("03/10/1982");  

string message = ValidateDate(dob);

lbldatemessage.Visible = !StringIsNullOrWhitespace(message);
lbldatemessage.Text = message ?? ""; //Ternary if message is null then default to empty string

Recuerde que puede formatear el mensaje como desee.

Respondido el 02 de enero de 18 a las 09:01

¿Qué tal esta solución?

static string CalcAge(DateTime birthDay)
{
    DateTime currentDate = DateTime.Now;         
    int approximateAge = currentDate.Year - birthDay.Year;
    int daysToNextBirthDay = (birthDay.Month * 30 + birthDay.Day) - 
        (currentDate.Month * 30 + currentDate.Day) ;

    if (approximateAge == 0 || approximateAge == 1)
    {                
        int month =  Math.Abs(daysToNextBirthDay / 30);
        int days = Math.Abs(daysToNextBirthDay % 30);

        if (month == 0)
            return "Your age is: " + daysToNextBirthDay + " days";

        return "Your age is: " + month + " months and " + days + " days"; ;
    }

    if (daysToNextBirthDay > 0)
        return "Your age is: " + --approximateAge + " Years";

    return "Your age is: " + approximateAge + " Years"; ;
}

Respondido 07 Feb 16, 00:02

private int GetAge(int _year, int _month, int _day
{
    DateTime yourBirthDate= new DateTime(_year, _month, _day);

    DateTime todaysDateTime = DateTime.Today;
    int noOfYears = todaysDateTime.Year - yourBirthDate.Year;

    if (DateTime.Now.Month < yourBirthDate.Month ||
        (DateTime.Now.Month == yourBirthDate.Month && DateTime.Now.Day < yourBirthDate.Day))
    {
        noOfYears--;
    }

    return  noOfYears;
}

Respondido 07 Feb 16, 00:02

El siguiente enfoque (extracto de Biblioteca de períodos de tiempo para .NET clase Datediff) considera el calendario de la información cultural:

// ----------------------------------------------------------------------
private static int YearDiff( DateTime date1, DateTime date2 )
{
  return YearDiff( date1, date2, DateTimeFormatInfo.CurrentInfo.Calendar );
} // YearDiff

// ----------------------------------------------------------------------
private static int YearDiff( DateTime date1, DateTime date2, Calendar calendar )
{
  if ( date1.Equals( date2 ) )
  {
    return 0;
  }

  int year1 = calendar.GetYear( date1 );
  int month1 = calendar.GetMonth( date1 );
  int year2 = calendar.GetYear( date2 );
  int month2 = calendar.GetMonth( date2 );

  // find the the day to compare
  int compareDay = date2.Day;
  int compareDaysPerMonth = calendar.GetDaysInMonth( year1, month1 );
  if ( compareDay > compareDaysPerMonth )
  {
    compareDay = compareDaysPerMonth;
  }

  // build the compare date
  DateTime compareDate = new DateTime( year1, month2, compareDay,
    date2.Hour, date2.Minute, date2.Second, date2.Millisecond );
  if ( date2 > date1 )
  {
    if ( compareDate < date1 )
    {
      compareDate = compareDate.AddYears( 1 );
    }
  }
  else
  {
    if ( compareDate > date1 )
    {
      compareDate = compareDate.AddYears( -1 );
    }
  }
  return year2 - calendar.GetYear( compareDate );
} // YearDiff

Uso:

// ----------------------------------------------------------------------
public void CalculateAgeSamples()
{
  PrintAge( new DateTime( 2000, 02, 29 ), new DateTime( 2009, 02, 28 ) );
  // > Birthdate=29.02.2000, Age at 28.02.2009 is 8 years
  PrintAge( new DateTime( 2000, 02, 29 ), new DateTime( 2012, 02, 28 ) );
  // > Birthdate=29.02.2000, Age at 28.02.2012 is 11 years
} // CalculateAgeSamples

// ----------------------------------------------------------------------
public void PrintAge( DateTime birthDate, DateTime moment )
{
  Console.WriteLine( "Birthdate={0:d}, Age at {1:d} is {2} years", birthDate, moment, YearDiff( birthDate, moment ) );
} // PrintAge

Respondido 05 Jul 18, 04:07

Esta pregunta clásica merece una Hora de Noda solución.

static int GetAge(LocalDate dateOfBirth)
{
    Instant now = SystemClock.Instance.Now;

    // The target time zone is important.
    // It should align with the *current physical location* of the person
    // you are talking about.  When the whereabouts of that person are unknown,
    // then you use the time zone of the person who is *asking* for the age.
    // The time zone of birth is irrelevant!

    DateTimeZone zone = DateTimeZoneProviders.Tzdb["America/New_York"];

    LocalDate today = now.InZone(zone).Date;

    Period period = Period.Between(dateOfBirth, today, PeriodUnits.Years);

    return (int) period.Years;
}

Uso:

LocalDate dateOfBirth = new LocalDate(1976, 8, 27);
int age = GetAge(dateOfBirth);

También te pueden interesar las siguientes mejoras:

  • Pasando en el reloj como un IClock, En lugar de usar SystemClock.Instance, mejoraría la capacidad de prueba.

  • Es probable que la zona horaria objetivo cambie, por lo que querrá un DateTimeZone parámetro también.

Vea también mi publicación de blog sobre este tema: Manejo de cumpleaños y otros aniversarios

Respondido 05 Jul 18, 04:07

Utilicé la solución de ScArcher2 para un cálculo exacto del año de la edad de una persona, pero necesitaba ir más allá y calcular sus meses y días junto con los años.

    public static Dictionary<string,int> CurrentAgeInYearsMonthsDays(DateTime? ndtBirthDate, DateTime? ndtReferralDate)
    {
        //----------------------------------------------------------------------
        // Can't determine age if we don't have a dates.
        //----------------------------------------------------------------------
        if (ndtBirthDate == null) return null;
        if (ndtReferralDate == null) return null;

        DateTime dtBirthDate = Convert.ToDateTime(ndtBirthDate);
        DateTime dtReferralDate = Convert.ToDateTime(ndtReferralDate);

        //----------------------------------------------------------------------
        // Create our Variables
        //----------------------------------------------------------------------
        Dictionary<string, int> dYMD = new Dictionary<string,int>();
        int iNowDate, iBirthDate, iYears, iMonths, iDays;
        string sDif = "";

        //----------------------------------------------------------------------
        // Store off current date/time and DOB into local variables
        //---------------------------------------------------------------------- 
        iNowDate = int.Parse(dtReferralDate.ToString("yyyyMMdd"));
        iBirthDate = int.Parse(dtBirthDate.ToString("yyyyMMdd"));

        //----------------------------------------------------------------------
        // Calculate Years
        //----------------------------------------------------------------------
        sDif = (iNowDate - iBirthDate).ToString();
        iYears = int.Parse(sDif.Substring(0, sDif.Length - 4));

        //----------------------------------------------------------------------
        // Store Years in Return Value
        //----------------------------------------------------------------------
        dYMD.Add("Years", iYears);

        //----------------------------------------------------------------------
        // Calculate Months
        //----------------------------------------------------------------------
        if (dtBirthDate.Month > dtReferralDate.Month)
            iMonths = 12 - dtBirthDate.Month + dtReferralDate.Month - 1;
        else
            iMonths = dtBirthDate.Month - dtReferralDate.Month;

        //----------------------------------------------------------------------
        // Store Months in Return Value
        //----------------------------------------------------------------------
        dYMD.Add("Months", iMonths);

        //----------------------------------------------------------------------
        // Calculate Remaining Days
        //----------------------------------------------------------------------
        if (dtBirthDate.Day > dtReferralDate.Day)
            //Logic: Figure out the days in month previous to the current month, or the admitted month.
            //       Subtract the birthday from the total days which will give us how many days the person has lived since their birthdate day the previous month.
            //       then take the referral date and simply add the number of days the person has lived this month.

            //If referral date is january, we need to go back to the following year's December to get the days in that month.
            if (dtReferralDate.Month == 1)
                iDays = DateTime.DaysInMonth(dtReferralDate.Year - 1, 12) - dtBirthDate.Day + dtReferralDate.Day;       
            else
                iDays = DateTime.DaysInMonth(dtReferralDate.Year, dtReferralDate.Month - 1) - dtBirthDate.Day + dtReferralDate.Day;       
        else
            iDays = dtReferralDate.Day - dtBirthDate.Day;             

        //----------------------------------------------------------------------
        // Store Days in Return Value
        //----------------------------------------------------------------------
        dYMD.Add("Days", iDays);

        return dYMD;
}

Respondido 14 Oct 12, 13:10

Versión SQL:

declare @dd smalldatetime = '1980-04-01'
declare @age int = YEAR(GETDATE())-YEAR(@dd)
if (@dd> DATEADD(YYYY, -@age, GETDATE())) set @age = @age -1

print @age  

Respondido el 30 de junio de 16 a las 12:06

He hecho un pequeño cambio en De Mark Soen respuesta: He reescrito la tercera línea para que la expresión se pueda analizar un poco más fácilmente.

public int AgeInYears(DateTime bday)
{
    DateTime now = DateTime.Today;
    int age = now.Year - bday.Year;            
    if (bday.AddYears(age) > now) 
        age--;
    return age;
}

También lo convertí en una función en aras de la claridad.

Respondido 22 Feb 18, 16:02

No es la respuesta que estás buscando? Examinar otras preguntas etiquetadas or haz tu propia pregunta.