Dat un "DateTime" reprezintă o persoană's ziua, cum pot calcula vârsta în ani?
Un ușor de înțeles și simplu de soluție.
// 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 year
if (birthdate.Date > today.AddYears(-age)) age--;
Cu toate acestea, acest lucru presupune că sunteți în căutarea pentru vest idee de vârstă și nu se utilizează Asia de Est socotelilor.
Acesta este un mod ciudat de a face, dar dacă ai format data "aaaallzz" și scade la data de naștere de la data curentă, apoi picătură ultimele 4 cifre'am ajuns la varsta :)
Eu nu't știu C#, dar cred că acest lucru va lucra în orice limbă.
20080814 - 19800703 = 280111
Drop ultimele 4 cifre = 28
.
C# Cod:
int now = int.Parse(DateTime.Now.ToString("yyyyMMdd"));
int dob = int.Parse(dateOfBirth.ToString("yyyyMMdd"));
int age = (now - dob) / 10000;
Sau, alternativ, fără conversie de tip în formă de o metodă de prelungire. Eroare de verificare omise:
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;
}
Eu nu't știu cum greșit soluție poate fi acceptată. Corect C# fragment a fost scris de Michael Stum
Aici este un test fragment:
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
Aici aveți metode:
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;
}
Eu nu't cred că vreunul dintre răspunsurile de până acum oferim pentru culturile care se calculează vârsta în mod diferit. A se vedea, de exemplu, din Asia de Est Age Origins față de cea din Occident.
Orice real răspunsul trebuie să includă localizare. De Model de Strategie ar fi, probabil, în ordine în acest exemplu.
Răspunsul simplu la această este de a aplica AddYears
așa cum se arată mai jos, pentru că aceasta este singura metodă nativă de a adăuga ani la 29 Feb. de ani bisecți și de a obține rezultatul corect din 28 Feb. pentru frecvente de ani.
Unii simt că 1 Mar. este ziua de naștere a leaplings, dar nici .Net, nici oficial statul sprijină acest lucru, nici nu logică comună explica de ce unele născut în februarie ar trebui să aibă 75% din zilele lor de naștere într-o altă lună.
Mai mult, o Vârstă metodă se pretează pentru a fi adăugate ca o extensie a "DateTime". Prin aceasta, puteți obține vârsta în cel mai simplu mod posibil:
int varsta = data nasterii.Vârsta();
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;
}
}
Acum, rula acest test:
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();
}
}
De data critică exemplu este aceasta:
Data nașterii: 2000-02-29 mai Târziu data: 2011-02-28 Varsta: 11
Ieșire:
{
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
}
Și pentru mai târziu de data 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
}
O altă funcție, nu de mine dar nu a găsit pe web și rafinat un pic:
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;
}
Doar două lucruri care vin în minte: Ce despre oamenii din țările care nu folosesc calendarul gregorian? DateTime.Acum este în server-cultură specifică cred. Nu am absolut 0 cunoștințe despre lucrul cu Asiatice calendare și nu știu dacă există o modalitate ușoară de a converti datele între calendare, dar doar în cazul în care're întrebați despre cei chinezi din anul 4660 :-)
2 Principalele probleme de rezolvat sunt:
1. Calcula vârsta Exactă - în ani, luni, zile, etc.
2. Se calculează, în General, percepută de vârstă - de obicei oamenii nu-mi pasă câți ani au, mai exact, le pasă doar atunci când ziua lor de naștere din anul curent este.
Soluție pentru 1 este evident:
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
Soluție pentru 2 este una care nu este atât de precis în total de stabilire a vârstei, dar este perceput la fel de precis de către oameni. De asemenea, oamenii folosesc, de obicei, atunci când se calculează vârsta lor "manual":
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
}
Note la 2.:
Doar o notă mai multe ... mi-ar crea 2 static metode supraîncărcate pentru ea, unul pentru utilizare universală, în al doilea rând pentru utilizarea în utilizare:
public static int GetAge(DateTime bithDay, DateTime today)
{
//chosen solution method body
}
public static int GetAge(DateTime birthDay)
{
return GetAge(birthDay, DateTime.Now);
}
Aceasta este versiunea vom folosi aici. Acesta funcționează, și-l's destul de simplu. L's aceeasi idee ca Jeff's, dar cred că's un pic mai clar, pentru că separă logica pentru scăderea unul, ca's un pic mai ușor de înțeles.
public static int GetAge(this DateTime dateOfBirth, DateTime dateAsAt)
{
return dateAsAt.Year - dateOfBirth.Year - (dateOfBirth.DayOfYear < dateAsAt.DayOfYear ? 0 : 1);
}
Ai putea extinde operatorul ternar pentru a face și mai clară, dacă credeți că acest lucru este neclar.
Evident, acest lucru este făcut ca o metodă de prelungire pe "DateTime", dar în mod clar te poti apuca de o linie de cod care functioneaza si pune-l oriunde. Aici avem o suprasarcină de metoda de Extensie care trece în `DateTime.Acum, doar pentru completare.
Eu folosesc asta:
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;
}
}
Acest lucru vă oferă "mai multe detalii" la această întrebare. Poate că acest lucru este ceea ce ai're în căutarea pentru
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);
Am creat un SQL Server-Funcții Definite de Utilizator pentru a calcula cineva's de vârstă, având în vedere data nașterii lor. Acest lucru este util atunci când aveți nevoie de ea, ca parte a unei interogări:
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;
}
};
Aici's încă un răspuns:
public static int AgeInYears(DateTime birthday, DateTime today)
{
return ((today.Year - birthday.Year) * 372 + (today.Month - birthday.Month) * 31 + (today.Day - birthday.Day)) / 372;
}
Acest lucru a fost intens testat unitar. E un pic "magic". Numărul 372 numărul de zile în care ar exista într-un an, dacă în fiecare lună a avut 31 de zile.
Explicația de ce functioneaza (a ridicat de aici) este:
Sa's set
Yn = DateTime.Acum.An, Yb = ziua de naștere.An, Mn = DateTime.Acum.Luni, Mb = ziua de naștere.Luna, Dn = DateTime.Acum.Zi, Db = ziua de naștere.Zi
vârsta = Yn - Yb + (31*(Mn - Mb) + (Dn - Db)) / 372`
știm că ceea ce avem nevoie este fie
Yn-Yb
dacă data a fost deja atins,Yn-Yb-1
dacă nu are.
o) Dacă Mn<Mb
, am
-341 <= 31*(Mn-Mo) <= -31 și -30 <= Dn-Db <= 30`
-371 <= 31*(Mn - Mb) + (Dn - Db) <= -1
Cu întreg divizia
(31*(Mn - Mb) + (Dn - Db)) / 372 = -1
b) Dacă Mn=Mb " și " Dn<Db
, am
31*(Mn - Mb) = 0 și -30 <= Dn-Db <= -1`
Cu integer division, din nou
(31*(Mn - Mb) + (Dn - Db)) / 372 = -1
c) Dacă Mn>Mb
, am
31 <= 31*(Mn-Mo) <= 341 și -30 <= Dn-Db <= 30`
1 <= 31*(Mn - Mb) + (Dn - Db) <= 371
Cu întreg divizia
(31*(Mn - Mb) + (Dn - Db)) / 372 = 0
d) Dacă Mn=Mb " și " Dn>Db
, am
31*(Mn - Mb) = 0 și 1 <= Dn-Db <= 3`0
Cu integer division, din nou
(31*(Mn - Mb) + (Dn - Db)) / 372 = 0
e) Dacă Mn=Mb " și " Dn=Db
, am
31*(Mn - Mb) + Dn-Db = 0`
și, prin urmare,
(31*(Mn - Mb) + (Dn - Db)) / 372 = 0
Am'am lucrat ceva timp la acest lucru și a venit cu acest lucru pentru a calcula cineva's de vârsta în ani, luni și zile. Am'am testat împotriva Feb 29 problemă și ani bisecți și se pare că funcționează, am'd aprecia orice feedback-ul:
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++;
}
}
Avem nevoie să ia în considerare persoanele care este mai mică de 1 an? ca și cultura Chineză, vom descrie copii mici' varsta de 2 luni de zile sau 4 săptămâni.
Mai jos este punerea în aplicare a mea, nu este la fel de simplu ca ceea ce mi-am imaginat, mai ales să se ocupe data ca 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";
}
}
Această punere în aplicare a trecut de mai jos cazurile de testare.
[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);
}
Sper's de ajutor.
Păstrarea este simplu (și, eventual, de prost:)).
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.");
TimeSpan diff = DateTime.Now - birthdayDateTime;
string age = String.Format("{0:%y} years, {0:%M} months, {0:%d}, days old", diff);
Am'm nu sunt sigur exact cum ai'd ca a revenit la tine, așa că am făcut un șir care poate fi citit.
Cel mai simplu mod de am'am găsit-o este aceasta. Acesta funcționează corect pentru SUA și europa de vest sale. Poate't vorbesc cu alte localizări, în special în locuri cum ar fi China. 4 suplimentar compară, cel mult, în urma inițială de calcul de vârstă.
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 ;
}
M-am uitat peste raspunsurile la acest lucru și am observat că nimeni nu a făcut referire la reglementare/legale implicațiile salt zi de nașteri. De exemplu, pe Wikipedia, daca're-a născut pe 29 februarie în diferite jurisdicții, ai're an bisect ziua variază:
Și cât de aproape am putea spune, la NOI, legile sunt tăcut în această privință, lăsând drept comun și de modul în care diferite organisme de reglementare a defini lucrurile în regulamentele lor.
În acest scop, o îmbunătățire:
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;
}
Ar trebui remarcat faptul că acest cod presupune:
O - vest (Europene) luarea în considerare a vârstei, și