Comment calculer l'âge de Quelqu'un en C#?
avec un DateTime
représentant l'anniversaire d'une personne, comment calculer son âge en années?
30 réponses
une solution facile à comprendre et 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 year
if (birthdate > today.AddYears(-age)) age--;
cependant, cela suppose que vous êtes à la recherche de la occidentale idée de l'âge et de ne pas utiliser East Asian reckoning .
c'est une façon étrange de le faire, mais si vous formatez la date à yyyymmdd
et soustrayez la date de naissance de la date actuelle puis laisser tomber les 4 derniers chiffres, vous avez l'âge:)
Je ne sais pas C#, mais je crois que ça marchera dans n'importe quelle langue.
20080814 - 19800703 = 280111
supprimer les 4 derniers chiffres = 28
.
C # Code:
int now = int.Parse(DateTime.Now.ToString("yyyyMMdd"));
int dob = int.Parse(dateOfBirth.ToString("yyyyMMdd"));
int age = (now - dob) / 10000;
ou alternativement sans conversion de type forme d'une méthode d'extension. Vérification des erreurs 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;
}
je ne sais pas comment la mauvaise solution peut être acceptée. Le bon c# snippet a été écrit par Michael Stum
voici un extrait d'essai:
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
Voici les méthodes:
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;
}
Je ne pense pas qu'aucune des réponses fournies jusqu'à présent ne tienne compte des cultures qui calculent l'âge différemment. Voir, par exemple, de l'Asie de l'Âge Reckoning et que dans l'Ouest.
Tout réel la réponse doit inclure la localisation. Le schéma de stratégie serait probablement en ordre dans cet exemple.
la réponse simple à cela est d'appliquer AddYears
comme indiqué ci-dessous parce que c'est la seule méthode native pour ajouter des années au 29 février. des années bissextiles, et obtenir le résultat correct du 28 Fév. pour les années communes.
certains pensent que le 1er mars est l'anniversaire des sauts, mais ni.net, ni aucune règle officielle ne le soutient, pas plus que la logique courante n'explique pourquoi certains nés en février devraient avoir 75% de leurs anniversaires dans un autre mois.
de plus, une méthode fondée sur l'âge se prête à être ajoutée en tant qu'extension à DateTime
. Par cela vous pouvez obtenir l'âge de la manière la plus simple possible:
- élément de la Liste
int age = date de naissance.Âge ();
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;
}
}
maintenant, exécutez ce 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();
}
}
l'exemple de la date critique est le suivant:
Date de naissance: 2000-02-29 plus tard date: 2011-02-28 âge: 11
sortie:
{
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
}
et pour la date ultérieure 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
}
Ma suggestion
int age = (int) ((DateTime.Now - bday).TotalDays/365.242199);
qui semble avoir l'année changée à la bonne date. (Je place l'essai jusqu'à l'âge de 107 ans)
une autre fonction, pas par moi mais trouvé sur le web et raffiné un peu:
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;
}
juste deux choses qui me viennent à l'esprit: Qu'en est-il des gens de pays qui n'utilisent pas le calendrier grégorien? DateTime.Est maintenant dans la culture spécifique au serveur je pense. J'ai absolument aucune connaissance sur le fait de travailler avec des calendriers asiatiques et je ne sais pas s'il y a un moyen facile de convertir les dates entre les calendriers, mais juste au cas où vous vous demandez à propos de ces les chinois de l'année 4660: -)
je suis en retard à la fête, Mais voici un one-liner:
int age = new DateTime(DateTime.Now.Subtract(birthday).Ticks).Year-1;
2 principaux problèmes à résoudre sont:
1. Calculer L'âge Exact - en années, mois, jours, etc.
2. Calcul généralement perçue âge - les gens ne se soucient généralement pas de l'âge qu'ils sont exactement, ils se soucient juste quand leur anniversaire dans l'année en cours est.
Solution pour 1 est évident:
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
Solution 2 est celui qui n'est pas aussi précis dans la détermination de total âge, mais qui est perçu comme le précise par les gens. Les gens aussi l'utilisent habituellement, quand ils calculent leur âge "manuellement":
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
}
les Notes 2.:
- C'est ma solution préférée
- nous ne pouvons pas utiliser DateTime.Nombre de jours dans les années bissextiles
- j'ai mis là peu plus de lignes pour la lisibilité
encore une note ... Je créerais 2 méthodes statiques surchargées pour cela, une pour l'usage universel, la deuxième pour l'usage-convivialité:
public static int GetAge(DateTime bithDay, DateTime today)
{
//chosen solution method body
}
public static int GetAge(DateTime birthDay)
{
return GetAge(birthDay, DateTime.Now);
}
il y a de nombreuses années, pour fournir un calculateur d'âge gimmick sur mon site web, j'ai écrit une fonction pour calculer l'âge à une fraction. C'est un port rapide de cette fonction à C# (de la version PHP ). J'ai bien peur de ne pas avoir été capable de tester la version C#, mais j'espère que vous apprécierez tout de même!
(il est vrai que C'est un peu compliqué dans le but de montrer les profils des utilisateurs sur le débordement de la pile, mais peut-être que les lecteurs trouveront utiliser pour cela. :- ))
double AgeDiff(DateTime date1, DateTime date2) {
double years = date2.Year - date1.Year;
/*
* If date2 and date1 + round(date2 - date1) are on different sides
* of 29 February, then our partial year is considered to have 366
* days total, otherwise it's 365. Note that 59 is the day number
* of 29 Feb.
*/
double fraction = 365
+ (DateTime.IsLeapYear(date2.Year) && date2.DayOfYear >= 59
&& (date1.DayOfYear < 59 || date1.DayOfYear > date2.DayOfYear)
? 1 : 0);
/*
* The only really nontrivial case is if date1 is in a leap year,
* and date2 is not. So let's handle the others first.
*/
if (DateTime.IsLeapYear(date2.Year) == DateTime.IsLeapYear(date1.Year))
return years + (date2.DayOfYear - date1.DayOfYear) / fraction;
/*
* If date2 is in a leap year, but date1 is not and is March or
* beyond, shift up by a day.
*/
if (DateTime.IsLeapYear(date2.Year)) {
return years + (date2.DayOfYear - date1.DayOfYear
- (date1.DayOfYear >= 59 ? 1 : 0)) / fraction;
}
/*
* If date1 is not on 29 February, shift down date1 by a day if
* March or later. Proceed normally.
*/
if (date1.DayOfYear != 59) {
return years + (date2.DayOfYear - date1.DayOfYear
+ (date1.DayOfYear > 59 ? 1 : 0)) / fraction;
}
/*
* Okay, here date1 is on 29 February, and date2 is not on a leap
* year. What to do now? On 28 Feb in date2's year, the ``age''
* should be just shy of a whole number, and on 1 Mar should be
* just over. Perhaps the easiest way is to a point halfway
* between those two: 58.5.
*/
return years + (date2.DayOfYear - 58.5) / fraction;
}
C'est la version que nous utilisons ici. Il fonctionne, et c'est assez simple. C'est la même idée que celle de Jeff mais je pense que c'est un peu plus clair parce que ça sépare la logique pour en soustraire une, donc c'est un peu plus facile à comprendre.
public static int GetAge(this DateTime dateOfBirth, DateTime dateAsAt)
{
return dateAsAt.Year - dateOfBirth.Year - (dateOfBirth.DayOfYear < dateAsAt.DayOfYear ? 0 : 1);
}
vous pourriez étendre l'opérateur ternaire pour le rendre encore plus clair, si vous pensez que ce genre de chose est imprécis.
évidemment ceci est fait comme une méthode d'extension sur DateTime
, mais clairement vous pouvez prenez cette ligne de code qui fait le travail et mettez-le n'importe où. Ici , nous avons une autre surcharge de la méthode D'Extension qui passe dans DateTime.Now
, juste pour l'exhaustivité.
j'utilise ceci:
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;
}
}
la meilleure façon que je connaisse à cause des années bissextiles et tout est:
DateTime birthDate = new DateTime(2000,3,1);
int age = (int)Math.Floor((DateTime.Now - birthDate).TotalDays / 365.25D);
Espérons que cette aide.
Cela donne "plus de détails" à cette question. Peut-être que c'est ce que vous cherchez
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);
j'ai créé une fonction SQL Server définie par L'utilisateur pour calculer l'âge de quelqu'un, étant donné sa date de naissance. Ceci est utile lorsque vous en avez besoin dans le cadre d'une requête:
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;
}
};
j'ai passé du temps à travailler là-dessus et j'ai trouvé ça pour calculer l'âge de quelqu'un en années, mois et jours. J'ai testé contre le problème du 29 février et les années bissextiles et cela semble fonctionner, j'apprécierais tout commentaire:
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++;
}
}
Voici encore une autre réponse:
public static int AgeInYears(DateTime birthday, DateTime today)
{
return ((today.Year - birthday.Year) * 372 + (today.Month - birthday.Month) * 31 + (today.Day - birthday.Day)) / 372;
}
ce produit a fait l'objet de nombreux essais à l'unité. Il ressemble un peu "magique". Le nombre 372 est le nombre de jours qu'il y aurait dans une année si chaque mois avait 31 jours.
l'explication de pourquoi cela fonctionne ( levé d'ici ) est:
faisons ensemble
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
nous savons que ce dont nous avons besoin est soit
Yn-Yb
si la date a déjà été atteinte,Yn-Yb-1
si elle n'a pas.a) Si
Mn<Mb
, nous avons-341 <= 31*(Mn-Mb) <= -31 and -30 <= Dn-Db <= 30
-371 <= 31*(Mn - Mb) + (Dn - Db) <= -1
avec une division entière
(31*(Mn - Mb) + (Dn - Db)) / 372 = -1
b) Si
Mn=Mb
etDn<Db
, nous avons31*(Mn - Mb) = 0 and -30 <= Dn-Db <= -1
avec nombre entier division, de nouveau
(31*(Mn - Mb) + (Dn - Db)) / 372 = -1
C) Si
Mn>Mb
, nous avons31 <= 31*(Mn-Mb) <= 341 and -30 <= Dn-Db <= 30
1 <= 31*(Mn - Mb) + (Dn - Db) <= 371
avec une division entière
(31*(Mn - Mb) + (Dn - Db)) / 372 = 0
D) Si
Mn=Mb
etDn>Db
, nous avons31*(Mn - Mb) = 0 and 1 <= Dn-Db <= 3
0avec division entière, à nouveau
(31*(Mn - Mb) + (Dn - Db)) / 372 = 0
e) si
Mn=Mb
etDn=Db
, nous avons31*(Mn - Mb) + Dn-Db = 0
et donc
(31*(Mn - Mb) + (Dn - Db)) / 372 = 0
Garder les choses simples (et peut-être stupide:)).
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);
Je ne sais pas comment exactement vous aimeriez qu'il vous soit retourné, donc j'ai juste fait une chaîne lisible.
devons-nous tenir compte des personnes de moins d'un an? comme culture chinoise, nous décrivons l'âge des petits bébés comme 2 mois ou 4 semaines.
ci-dessous est ma mise en œuvre, il n'est pas aussi simple que ce que j'ai imaginé, surtout pour traiter date comme 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";
}
}
cette implémentation a passé les cas de test ci-dessous.
[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);
}
J'espère que c'est utile.
la façon la plus simple que j'ai jamais trouvée est celle-ci. Il fonctionne correctement pour les États-Unis et les pays d'europe occidentale. Je ne peux pas parler à d'autres endroits, surtout en Chine. 4 extra compare, tout au plus, après le calcul initial de l'âge.
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 ;
}
je regardais les réponses à cela et j'ai remarqué que personne n'a fait référence aux implications réglementaires/juridiques des naissances leap day. Par exemple, par Wikipedia , si vous êtes né le 29 février dans diverses juridictions, vous êtes année non bissextile anniversaire varie:
- au Royaume-Uni et à Hong Kong: c'est le jour ordinal de l'année, donc le lendemain, le 1er Mars est votre anniversaire.
- en Nouvelle-Zélande: c'est la veille, le 28 février pour le permis de conduire, et le 1er mars pour d'autres raisons.
- Taiwan: on est le 28 février.
et aussi près que I aux États-Unis, les lois sont muettes sur la question, laissant le soin à la common law et à la façon dont les divers organismes de réglementation définissent les choses dans leurs règlements.
à cette fin, une amélioration:
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;
}
il est à noter que ce code suppose:
- occidentale (Européenne) prise en compte de l'âge, et
- Un calendrier, comme le calendrier Grégorien qui insère un saut seul jour à la fin d'un mois.
C'est l'une des réponses les plus précises qui est capable de résoudre l'anniversaire du 29 février comparer à n'importe quelle année du 28 février.
public int GetAge(DateTime birthDate)
{
int age = DateTime.Now.Year - birthDate.Year;
if (birthDate.DayOfYear > DateTime.Now.DayOfYear)
age--;
return age;
}
Voici une solution.
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);
ce n'est pas une réponse directe, mais plutôt un raisonnement philosophique sur le problème en question d'un point de vue quasi-scientifique.
je dirais que la question ne spécifie pas l'unité ni la culture dans laquelle mesurer l'âge, la plupart des réponses semblent supposer une représentation annuelle entière. L'unité SI pour le temps est second
, donc la réponse générique correcte devrait être (bien sûr en supposant normalisé DateTime
et en ne tenant aucun compte de quelque effets relativistes):
var lifeInSeconds = (DateTime.Now.Ticks - then.Ticks)/TickFactor;
dans la manière chrétienne de calculer l'âge en années:
var then = ... // Then, in this case the birthday
var now = DateTime.UtcNow;
int age = now.Year - then.Year;
if (now.AddYears(-age) < then) age--;
dans le domaine de la finance , il existe un problème similaire lorsque l'on calcule ce que l'on appelle souvent la Fraction du nombre de jours , qui correspond à peu près à un nombre d'années pour une période donnée. Et la question de l'âge est vraiment une question de mesure du temps.
exemple pour le réel / actuel (en comptant tous les jours "correctement)") convention:
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;
une autre façon assez courante de mesurer le temps en général est de "sérialiser" (le gars qui a nommé cette convention de date doit sérieusement avoir été trippin'):
DateTime start, end = .... // Whatever, assume start is before end
int days = (end - start).Days;
je me demande combien de temps nous devons passer avant qu'un âge relativiste en secondes devienne plus utile que l'approximation grossière des cycles de la Terre autour du soleil au cours de sa vie jusqu'à présent:) ou en d'autres termes, quand une période doit être donnée un emplacement ou une fonction représentant le mouvement pour lui-même pour être valide :)
Que pensez-vous de cette solution?
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"; ;
}
j'ai une méthode personnalisée pour calculer l'âge, plus un message de validation de bonus juste au cas où il aide:
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
}
méthode appeler ici et transmettre la valeur de datetime (MM/JJ/yyyy si le serveur est défini à USA locale). Remplacez ceci par quelque chose d'une boîte à messages ou n'importe quel Contenant à afficher:
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
rappelez-vous que vous pouvez formater le message comme vous le souhaitez.
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;
}
j'ai utilisé la solution de ScArcher2 pour un calcul précis de l'année d'une personne de l'âge, mais j'ai dû aller plus loin et de calculer leurs mois et jours avec les années.
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;
}
version 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
j'ai fait un petit changement à réponse de Mark Soen : j'ai réécrit la troisième ligne de sorte que l'expression peut être analysée un peu plus facilement.
public int AgeInYears(DateTime bday)
{
DateTime now = DateTime.Today;
int age = now.Year - bday.Year;
if (bday.AddYears(age) > now)
age--;
return age;
}
j'en ai fait une fonction pour plus de clarté.