lo que todas las respuestas hasta ahora se han perdido es que depende de dónde nació la persona y dónde se encuentra ahora.
Yaur
40
@Yaur: solo convierta el tiempo de ahora + nacimiento en 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 usar GeoLocating.
Si estamos teniendo en cuenta la sugerencia de @Yaur de los cálculos de zonas horarias cruzadas, ¿debería el horario de verano afectar el cálculo de alguna manera?
DDM
66
Voto negativo porque esta es claramente una pregunta de tarea y no se proporcionó ningún intento existente.
Marie
Respuestas:
2123
Una solución fácil de entender y simple.
// Save today's date.var today =DateTime.Today;// Calculate the age.var age = today.Year- birthdate.Year;// Go back to the year the person was born in case of a leap yearif(birthdate.Date> today.AddYears(-age)) age--;
Sin embargo, esto supone que está buscando la idea occidental de la edad y no está usando el cálculo del este asiático .
Solo quería comentar sobre el rendimiento de DateTime.Now. Si no necesita un valor de zona horaria preciso, use DateTime.UtcNow es mucho más rápido.
JAG
104
Dado que estamos hablando de cumpleaños, puede usar DateTime. Hoy, dado el tiempo, la parte no tiene relevancia.
Tristan Warner-Smith
78
Esta respuesta no funciona con todas las configuraciones regionales y todas las edades. Varios países han omitido fechas después del nacimiento de las personas que viven actualmente, como Rusia (1918), Grecia (1924) y Turquía (1926).
Lars D
30
En realidad, todavía no es del todo correcto. Este código supone que 'bday' es la porción de fecha de un DateTime. Es un caso marginal (supongo que la mayoría de las personas simplemente pasarán fechas y no fechas y horas), pero si pasas en un cumpleaños como una fecha y hora donde el tiempo es mayor que 00:00:00, entonces ' Me encontraré con el error que Danvil señaló. Establecer bday = bday.Date corrige esto.
Øyvind el
119
La última línea me hizo pensar demasiado. En cambio, ¿qué tal: if (bday.AddYears (age)> now) age--; Esto parece ser una expresión más intuitiva.
cdiggins
1015
Esta es una forma extraña de hacerlo, pero si formatea la fecha yyyymmddy resta la fecha de nacimiento de la fecha actual, 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 un método de extensión. Error de comprobación omitido:
publicstaticInt32GetAge(thisDateTime 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;}
En realidad, esto es ideal para usar en MS-SQL con campos de fecha y hora (días totales desde 01-011900)
Patrik
55
@numerek Publique las modificaciones sugeridas como su propia respuesta. Por lo que vale, el año actual multiplicado por 10000 no está cerca de un desbordamiento de enteros, en dos órdenes de magnitud. 20,150,000 vs 2,147,483,648
GalacticCowboy
77
@LongChalk 20180101 - 20171231 = 8870. Suelta los últimos 4 dígitos y tienes (un implícito) 0para la edad. ¿Cómo se consigue 1?
Rufus L
44
Sé que esta es una respuesta anterior, pero no haría un método de extensión, no es el lugar adecuado para definir tales lógicas.
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 años no bisiestos, en lugar del 28 de febrero. En realidad, cualquiera de las opciones puede ser correcta . Wikipedia tiene algo que decir sobre esto . Entonces, si bien su código no es "incorrecto", tampoco lo es la solución aceptada.
Matt Johnson-Pint
18
@ MattJohnson Creo que eso es realmente correcto. Si mi cumpleaños fue el 29 de febrero, entonces el 28 de febrero mi cumpleaños no ha pasado, y aún debería tener la misma edad que el 27 de febrero. Sin embargo, el 1 de marzo, pasamos mi cumpleaños y debería tener la próxima edad. En los Estados Unidos, una empresa que vende alcohol tendrá un letrero que dice algo como "Si naciste después de este día en AAAA, no puedes comprar alcohol" (donde AAAA cambia todos los años). Eso significa que alguien nacido el 29 de febrero no puede comprar alcohol el 28 de febrero del año en que cumpla 21 años (la mayoría de los lugares), y apoya la idea de que no son un año mayores hasta el 1 de marzo
jfren484
44
@ jfren484 - lea el artículo de Wikipedia. Varía considerablemente entre jurisdicciones.
Matt Johnson-Pint
99
@ jfren484 Su reclamo no tiene absolutamente nada que ver con la filosofía; pero todo que ver con tu propio sentimiento personal . Cuando una persona nacida el 29 de febrero, las "edades" no son importantes 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 un permiso de conducir). Considere la edad para beber en los Estados Unidos (21 años): para la mayoría de las personas son 7670 días. Son 7671 días si nacen antes del 29 de febrero en el año bisiesto o del 1 de marzo antes del año bisiesto. Si nació el 29 de febrero: el 28 de febrero son 7670 días y el 1 de marzo son 7671 días. La elección es arbitraria , puede ir en cualquier dirección.
Desilusionado
44
@ CraigYoung No entiendes lo que quise decir filosóficamente. Usé ese término como contraste con legalmente. Si uno está escribiendo una solicitud que necesita conocer la edad legal de una persona, entonces todo lo que necesita saber es cómo las jurisdicciones legales en las que se usa su aplicación en / para tratar a las personas nacidas el 29 de febrero. Sin embargo, si estamos hablando de cómo debería tratarse, 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 discutir el 1 de marzo que el 28 de febrero.
jfren484
110
La respuesta simple a esto es aplicar AddYearscomo 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 leaplings, pero ni .Net ni ninguna regla oficial lo respaldan, 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 para ser agregado como una extensión de DateTime. Con esto, puede obtener la edad de la manera más simple posible:
Elemento de la lista
int age = birthDate.Age ();
publicstaticclassDateTimeExtensions{/// <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>publicstaticintAge(thisDateTime birthDate){returnAge(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>publicstaticintAge(thisDateTime 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, ejecuta esta prueba:
classProgram{staticvoidMain(string[] args){RunTest();}privatestaticvoidRunTest(){DateTime birthDate =newDateTime(2000,2,28);DateTime laterDate =newDateTime(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 la fecha crítica es este:
Fecha de nacimiento: 2000-02-29 Fecha posterior: 2011-02-28 Edad: 11
Un comentario con respecto a tener el cumpleaños del 29 de febrero el 1 de marzo, técnicamente, tenerlo el 28 es demasiado temprano (de hecho, 1 día antes). El 1 es un día demasiado tarde. Pero dado que el cumpleaños está entre, usar el 1 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 el 2 y 3) cada año, pero no el 28 de febrero.
CyberClaw
1
Desde un punto de diseño de software, escribir esto como un método de extensión no tiene mucho sentido para mí. date.Age(other)?
marsze
90
Mi sugerencia
int age =(int)((DateTime.Now- bday).TotalDays/365.242199);
Parece que el año está cambiando en la fecha correcta. (Detecté pruebas hasta los 107 años).
La duración promedio de un año en el calendario gregoriano es de 365.2425 días.
dan04
44
Yo diría que esta es una de las soluciones más simples y es lo suficientemente buena . A quién le importa si tengo 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 correcto, aunque no matemáticamente. Realmente me gusta esta solución.
Peter Perháč
13
^^ Porque a veces es importante. En mis pruebas, esto falla en el cumpleaños de las personas, les informa que son más jóvenes de lo que son.
ChadT
76
Otra función, no para mí, pero la encontré en la web y la refiné un poco:
publicstaticintGetAge(DateTime birthDate){DateTime n =DateTime.Now;// To avoid a race condition around midnightint 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 acerca de cómo trabajar realmente con calendarios asiáticos y no sé si hay una manera fácil de convertir fechas entre calendarios, pero en caso de que te preguntes sobre esos chinos del año 4660 :-)
Esto parece manejar mejor diferentes regiones (formatos de fecha).
webdad3
53
2 Los principales problemas a resolver son:
1. Calcular la edad exacta - en años, meses, días, etc.
2. Calcule la edad generalmente percibida : a las personas generalmente no les importa cuántos años tienen exactamente, solo les importa cuándo es su cumpleaños en el año en curso.
La solución para 1 es obvia:
DateTime birth =DateTime.Parse("1.1.2000");DateTime today =DateTime.Today;//we usually don't care about birth timeTimeSpan age = today - birth;//.NET FCL should guarantee this as precisedouble ageInDays = age.TotalDays;//total number of days ... also precisedouble daysInYear =365.2425;//statistical value for 400 yearsdouble ageInYears = ageInDays / daysInYear;//can be shifted ... not so precise
La solución para 2 es la que no es tan precisa para determinar la edad total, pero es percibida como precisa por las personas. Las personas también suelen usarlo, 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 yearsif(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 estáticos sobrecargados para él, uno para uso universal, el segundo para facilidad de uso:
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 cumplir 14 años (1990-05-31).
Kjensen
43
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 una, por lo que es un poco más fácil de entender.
Podría ampliar el operador ternario para hacerlo aún más claro, si cree que ese tipo de cosas no está claro.
Obviamente, esto se hace como un método de extensión 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 de Extensión que pasa DateTime.Now, solo para completar.
Creo que esto puede estar apagado 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, debe hacer una comparación lexicográfica de pares (Month, DayOfMonth) y usar eso para el condicional.
Doug McClean
1
tampoco va a mostrar la edad correcta a partir de su cumpleaños.
dotjoe
43
La mejor manera que conozco debido a los años bisiestos y todo es:
DateTime birthDate =newDateTime(2000,3,1);int age =(int)Math.Floor((DateTime.Now- birthDate).TotalDays/365.25D);
Esto le da "más detalles" a esta pregunta. Quizás esto es lo que estás buscando
DateTime birth =newDateTime(1974,8,29);DateTime today =DateTime.Now;TimeSpan span = today - birth;DateTime age =DateTime.MinValue+ span;// Make adjustment due to MinValue equalling 1/1/1int 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 wellConsole.Write("{0} years, {1} months, {2} days", years, months, days);
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 utilizando las funciones AddYears (), AddMonths y AddDays (), no siempre devolverá Datetime .Ahora fecha.
Athanasios Kataras
3
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 lo que estás haciendo. He preguntado en los foros de Microsoft y Microsoft ha confirmado que tiene en cuenta los años bisiestos entre 2 fechas.
Jacqueline Loriault
2
Considere los siguientes DOS senarios. 1st DateTime.Now 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 senador DateTime.Now 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 intervalo de tiempo en un año no bisiesto. Si DateTime.MinValue fue un año bisiesto, los resultados serían 1 año en el primero y 0 años, 11 meses y 30 días. (Pruébalo en tu código).
Athanasios Kataras
1
¡Votación a favor! Se me ocurrió una solución que es bastante idéntica (usé DateTime.MinValue.AddTicks (span.Ticks) en lugar de +, pero el resultado es el mismo y el tuyo tiene algunos caracteres menos código).
Makotosan
44
Tienes toda la razón, no lo es. Pero si fuera así, ese sería el resultado. ¿Por qué eso importa? No lo hace. En cualquier caso, salto o no, entonces 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 el código y verá que tengo razón.
Athanasios Kataras
28
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;publicpartialclassUserDefinedFunctions{[SqlFunction(DataAccess=DataAccessKind.Read)]publicstaticSqlInt32CalculateAge(string strBirthDate){DateTime dtBirthDate =newDateTime();
dtBirthDate =Convert.ToDateTime(strBirthDate);DateTime dtToday =DateTime.Now;// get the difference in yearsint years = dtToday.Year- dtBirthDate.Year;// subtract another year if we're before the// birth day in the current yearif(dtToday.Month< dtBirthDate.Month||(dtToday.Month== dtBirthDate.Month&& dtToday.Day< dtBirthDate.Day))
years=years-1;int intCustomerAge = years;return intCustomerAge;}};
Esto ha sido ampliamente probado en unidades. Se ve 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.
Me topé con esta discusión larga y molesta, y su solución es un enfoque realmente agradable y pequeño. Gracias por hacerlo simple
nabuchodonossor
25
He pasado algún tiempo trabajando en esto y se me ocurrió esto para calcular la edad de alguien en años, meses y días. He probado el problema del 29 de febrero y los años bisiestos y parece funcionar, agradecería cualquier comentario:
publicvoidLoopAge(DateTime myDOB,DateTimeFutureDate){int years =0;int months =0;int days =0;DateTime tmpMyDOB =newDateTime(myDOB.Year, myDOB.Month,1);DateTime tmpFutureDate =newDateTime(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 dayif(DateTime.IsLeapYear(myDOB.Year)&& myDOB.Month==2&& myDOB.Day==29){//but only if the future date is less than 1st Marchif(FutureDate>=newDateTime(FutureDate.Year,3,1))
days++;}}
¿Debemos tener en cuenta 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 imaginé, especialmente para tratar con fechas como 2/28.
publicstaticstringHowOld(DateTime birthday,DateTime now){if(now < birthday)thrownewArgumentOutOfRangeException("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 weekDateTime 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");}}}elseif(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]publicvoidTestAge(){string age =HowOld(newDateTime(2011,1,1),newDateTime(2012,11,30));Assert.AreEqual("1 year", age);
age =HowOld(newDateTime(2011,11,30),newDateTime(2012,11,30));Assert.AreEqual("1 year", age);
age =HowOld(newDateTime(2001,1,1),newDateTime(2012,11,30));Assert.AreEqual("11 years", age);
age =HowOld(newDateTime(2012,1,1),newDateTime(2012,11,30));Assert.AreEqual("10 months", age);
age =HowOld(newDateTime(2011,12,1),newDateTime(2012,11,30));Assert.AreEqual("11 months", age);
age =HowOld(newDateTime(2012,10,1),newDateTime(2012,11,30));Assert.AreEqual("1 month", age);
age =HowOld(newDateTime(2008,2,28),newDateTime(2009,2,28));Assert.AreEqual("1 year", age);
age =HowOld(newDateTime(2008,3,28),newDateTime(2009,2,28));Assert.AreEqual("11 months", age);
age =HowOld(newDateTime(2008,3,28),newDateTime(2009,3,28));Assert.AreEqual("1 year", age);
age =HowOld(newDateTime(2009,1,28),newDateTime(2009,2,28));Assert.AreEqual("1 month", age);
age =HowOld(newDateTime(2009,2,1),newDateTime(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(newDateTime(2009,1,31),newDateTime(2009,2,28));Assert.AreEqual("4 weeks", age);
age =HowOld(newDateTime(2009,2,1),newDateTime(2009,2,28));Assert.AreEqual("3 weeks", age);
age =HowOld(newDateTime(2009,2,1),newDateTime(2009,3,1));Assert.AreEqual("1 month", age);
age =HowOld(newDateTime(2012,11,5),newDateTime(2012,11,30));Assert.AreEqual("3 weeks", age);
age =HowOld(newDateTime(2012,11,1),newDateTime(2012,11,30));Assert.AreEqual("4 weeks", age);
age =HowOld(newDateTime(2012,11,20),newDateTime(2012,11,30));Assert.AreEqual("1 week", age);
age =HowOld(newDateTime(2012,11,25),newDateTime(2012,11,30));Assert.AreEqual("5 days", age);
age =HowOld(newDateTime(2012,11,29),newDateTime(2012,11,30));Assert.AreEqual("1 day", age);
age =HowOld(newDateTime(2012,11,30),newDateTime(2012,11,30));Assert.AreEqual("just born", age);
age =HowOld(newDateTime(2000,2,29),newDateTime(2009,2,28));Assert.AreEqual("8 years", age);
age =HowOld(newDateTime(2000,2,29),newDateTime(2009,3,1));Assert.AreEqual("9 years", age);Exception e =null;try{
age =HowOld(newDateTime(2012,12,1),newDateTime(2012,11,30));}catch(ArgumentOutOfRangeException ex){
e = ex;}Assert.IsTrue(e !=null);}
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
19
La forma más simple que he encontrado es esta. Funciona correctamente para los entornos locales de EE. UU. Y Europa occidental. No puedo hablar con otros locales, especialmente lugares como China. 4 comparaciones adicionales, como máximo, después del cálculo inicial de la edad.
publicintAgeInYears(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);elseif(reference.Month< birth.Month)--years;else// in birth month{if(reference.Day< birth.Day)--years;}return years ;}
Estaba buscando las respuestas a esto y noté que nadie ha hecho referencia a las implicaciones legales / regulatorias de los nacimientos en días bisiestos. Por ejemplo, según Wikipedia , si naciste el 29 de febrero en varias jurisdicciones, el cumpleaños de tu año no bisiesto varía:
En el Reino Unido y Hong Kong: es el día ordinal del año, así 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 fines de 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 al respecto, dejándolo a la ley común y a cómo varios organismos reguladores definen las cosas en sus regulaciones.
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 casi científico.
Yo diría que la pregunta no especifica la unidad ni la cultura para 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 normalizadaDateTime y sin tener en cuenta 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 birthdayvar now =DateTime.UtcNow;int age = now.Year- then.Year;if(now.AddYears(-age)< then) age--;
En finanzas hay un problema similar al calcular algo que a menudo se denomina fracción de recuento de días , que es aproximadamente una cantidad de años para un período determinado. Y el problema de la edad es realmente un problema de medición del tiempo.
Ejemplo para la convención real / real (contando todos los días "correctamente"):
DateTime start, end =....// Whatever, assume start is before enddouble 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 mediante la "serialización" (el tipo que nombró esta convención de fecha debe haber estado tropezando seriamente):
DateTime start, end =....// Whatever, assume start is before endint days =(end - start).Days;
Me pregunto cuánto tiempo tenemos que pasar antes de que una edad 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 para que sea válida :)
Tengo un método personalizado para calcular la edad, además de un mensaje de validación de bonificación en caso de que ayude:
publicvoidGetAge(DateTime dob,DateTime now,outint years,outint months,outint days){
years =0;
months =0;
days =0;DateTime tmpdob =newDateTime(dob.Year, dob.Month,1);DateTime tmpnow =newDateTime(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 >=newDateTime(now.Year,3,1))
days++;}privatestringValidateDate(DateTime dob)//This method will validate the date{intYears=0;intMonths=0;intDays=0;GetAge(dob,DateTime.Now,outYears,outMonths,outDays);if(Years<18)
message =Years+" is too young. Please try again on your 18th birthday.";elseif(Years>=65)
message =Years+" is too old. Date of Birth must not be 65 or older.";elsereturnnull;//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 un cuadro de mensaje o cualquier contenedor para mostrar:
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 de la forma que desee.
staticstringCalcAge(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";;}
// ----------------------------------------------------------------------publicvoidCalculateAgeSamples(){PrintAge(newDateTime(2000,02,29),newDateTime(2009,02,28));// > Birthdate=29.02.2000, Age at 28.02.2009 is 8 yearsPrintAge(newDateTime(2000,02,29),newDateTime(2012,02,28));// > Birthdate=29.02.2000, Age at 28.02.2012 is 11 years}// CalculateAgeSamples// ----------------------------------------------------------------------publicvoidPrintAge(DateTime birthDate,DateTime moment ){Console.WriteLine("Birthdate={0:d}, Age at {1:d} is {2} years", birthDate, moment,YearDiff( birthDate, moment ));}// PrintAge
Esta clásica pregunta merece una solución Noda Time .
staticintGetAge(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 =newLocalDate(1976,8,27);int age =GetAge(dateOfBirth);
También te pueden interesar las siguientes mejoras:
Pasar el reloj como IClock, en lugar de usar SystemClock.Instance, mejoraría la capacidad de prueba.
La zona horaria objetivo probablemente cambiará, por lo que también querrá un DateTimeZoneparámetro.
He hecho contribuciones a él, pero es principalmente de Jon Skeet.
Matt Johnson-Pint
9
Utilicé la solución de ScArcher2 para un cálculo preciso del año de la edad de una persona, pero necesitaba llevarlo más lejos y calcular sus meses y días junto con los años.
publicstaticDictionary<string,int>CurrentAgeInYearsMonthsDays(DateTime? ndtBirthDate,DateTime? ndtReferralDate){//----------------------------------------------------------------------// Can't determine age if we don't have a dates.//----------------------------------------------------------------------if(ndtBirthDate ==null)returnnull;if(ndtReferralDate ==null)returnnull;DateTime dtBirthDate =Convert.ToDateTime(ndtBirthDate);DateTime dtReferralDate =Convert.ToDateTime(ndtReferralDate);//----------------------------------------------------------------------// Create our Variables//----------------------------------------------------------------------Dictionary<string,int> dYMD =newDictionary<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;}
He hecho un pequeño cambio en la respuesta de Mark Soen : he reescrito la tercera línea para que la expresión se pueda analizar un poco más fácilmente.
publicintAgeInYears(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.
Respuestas:
Una solución fácil de entender y simple.
Sin embargo, esto supone que está buscando la idea occidental de la edad y no está usando el cálculo del este asiático .
fuente
Esta es una forma extraña de hacerlo, pero si formatea la fecha
yyyymmdd
y resta la fecha de nacimiento de la fecha actual, suelte los últimos 4 dígitos que tiene la edad :)No sé C #, pero creo que esto funcionará en cualquier idioma.
Suelta los últimos 4 dígitos =
28
.Código C #:
O, alternativamente, sin toda la conversión de tipos en forma de un método de extensión. Error de comprobación omitido:
fuente
20180101 - 20171231 = 8870
. Suelta los últimos 4 dígitos y tienes (un implícito)0
para la edad. ¿Cómo se consigue1
?Aquí hay un fragmento de prueba:
Aquí tienes los métodos:
fuente
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 leaplings, pero ni .Net ni ninguna regla oficial lo respaldan, 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 para ser agregado como una extensión de
DateTime
. Con esto, puede obtener la edad de la manera más simple posible:int age = birthDate.Age ();
Ahora, ejecuta esta prueba:
El ejemplo de la fecha crítica es este:
Fecha de nacimiento: 2000-02-29 Fecha posterior: 2011-02-28 Edad: 11
Salida:
Y para la fecha posterior 28-02-28:
fuente
date.Age(other)
?Mi sugerencia
Parece que el año está cambiando en la fecha correcta. (Detecté pruebas hasta los 107 años).
fuente
days in a year = 365.242199
Otra función, no para mí, pero la encontré en la web y la refiné un poco:
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 acerca de cómo trabajar realmente con calendarios asiáticos y no sé si hay una manera fácil de convertir fechas entre calendarios, pero en caso de que te preguntes sobre esos chinos del año 4660 :-)
fuente
2 Los principales problemas a resolver son:
1. Calcular la edad exacta - en años, meses, días, etc.
2. Calcule la edad generalmente percibida : a las personas generalmente no les importa cuántos años tienen exactamente, solo les importa cuándo es su cumpleaños en el año en curso.
La solución para 1 es obvia:
La solución para 2 es la que no es tan precisa para determinar la edad total, pero es percibida como precisa por las personas. Las personas también suelen usarlo, cuando calculan su edad "manualmente":
Notas a 2 .:
Solo una nota más ... Crearía 2 métodos estáticos sobrecargados para él, uno para uso universal, el segundo para facilidad de uso:
fuente
Aquí hay una frase:
fuente
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 una, por lo que es un poco más fácil de entender.
Podría ampliar el operador ternario para hacerlo aún más claro, si cree que ese tipo de cosas no está claro.
Obviamente, esto se hace como un método de extensión
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 de Extensión que pasaDateTime.Now
, solo para completar.fuente
La mejor manera que conozco debido a los años bisiestos y todo es:
fuente
Yo uso esto:
fuente
Esto le da "más detalles" a esta pregunta. Quizás esto es lo que estás buscando
fuente
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:
fuente
Aquí hay otra respuesta:
Esto ha sido ampliamente probado en unidades. Se ve 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:
fuente
He pasado algún tiempo trabajando en esto y se me ocurrió esto para calcular la edad de alguien en años, meses y días. He probado el problema del 29 de febrero y los años bisiestos y parece funcionar, agradecería cualquier comentario:
fuente
¿Debemos tener en cuenta 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 imaginé, especialmente para tratar con fechas como 2/28.
Esta implementación ha pasado por debajo de los casos de prueba.
Espero que sea útil.
fuente
Manteniéndolo simple (y posiblemente estúpido :)).
fuente
La forma más simple que he encontrado es esta. Funciona correctamente para los entornos locales de EE. UU. Y Europa occidental. No puedo hablar con otros locales, especialmente lugares como China. 4 comparaciones adicionales, como máximo, después del cálculo inicial de la edad.
Estaba buscando las respuestas a esto y noté que nadie ha hecho referencia a las implicaciones legales / regulatorias de los nacimientos en días bisiestos. Por ejemplo, según Wikipedia , si naciste el 29 de febrero en varias jurisdicciones, el cumpleaños de tu año no bisiesto varía:
Y por lo que puedo decir, en los EE. UU., Los estatutos guardan silencio al respecto, dejándolo a la ley común y a cómo varios organismos reguladores definen las cosas en sus regulaciones.
Para ese fin, una mejora:
Cabe señalar que este código supone:
fuente
No estoy seguro de cómo exactamente le gustaría que le regresara, así que acabo de hacer una cadena legible.
fuente
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 casi científico.
Yo diría que la pregunta no especifica la unidad ni la cultura para 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 normalizadaDateTime
y sin tener en cuenta los efectos relativistas):En la forma cristiana de calcular la edad en años:
En finanzas hay un problema similar al calcular algo que a menudo se denomina fracción de recuento de días , que es aproximadamente una cantidad de años para un período determinado. Y el problema de la edad es realmente un problema de medición del tiempo.
Ejemplo para la convención real / real (contando todos los días "correctamente"):
Otra forma bastante común de medir el tiempo en general es mediante la "serialización" (el tipo que nombró esta convención de fecha debe haber estado tropezando seriamente):
Me pregunto cuánto tiempo tenemos que pasar antes de que una edad 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 para que sea válida :)
fuente
Aquí hay una solución.
fuente
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.
fuente
Tengo un método personalizado para calcular la edad, además de un mensaje de validación de bonificación en caso de que ayude:
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 un cuadro de mensaje o cualquier contenedor para mostrar:
Recuerde que puede formatear el mensaje de la forma que desee.
fuente
¿Qué tal esta solución?
fuente
fuente
El siguiente enfoque (extracto de Time Period Library para .NET class DateDiff ) considera el calendario de la información cultural:
Uso:
fuente
Esta clásica pregunta merece una solución Noda Time .
Uso:
También te pueden interesar las siguientes mejoras:
Pasar el reloj como
IClock
, en lugar de usarSystemClock.Instance
, mejoraría la capacidad de prueba.La zona horaria objetivo probablemente cambiará, por lo que también querrá un
DateTimeZone
parámetro.Vea también mi publicación de blog sobre este tema: Manejo de cumpleaños y otros aniversarios
fuente
Utilicé la solución de ScArcher2 para un cálculo preciso del año de la edad de una persona, pero necesitaba llevarlo más lejos y calcular sus meses y días junto con los años.
fuente
Versión SQL:
fuente
He hecho un pequeño cambio en la respuesta de Mark Soen : he reescrito la tercera línea para que la expresión se pueda analizar un poco más fácilmente.
También lo convertí en una función en aras de la claridad.
fuente