En C #, comment calculer l'âge d'une personne en fonction d'un anniversaire de type DateTime?

1869

Étant donné que DateTimel'anniversaire d'une personne représente, comment puis-je calculer son âge en années?

Shaik Raffi
la source
147
ce que toutes les réponses ont jusqu'ici manqué, c'est que cela dépend où la personne est née et où elle se trouve en ce moment.
Yaur
40
@Yaur: Convertissez simplement l'heure actuelle + la naissance en GMT / UTC, l'âge n'est qu'une valeur relative, les fuseaux horaires ne sont donc pas pertinents. Pour déterminer le fuseau horaire actuel de l'utilisateur, vous pouvez utiliser GeoLocating.
Stefan Steiger
Pourquoi ne pas envisager la [date julienne] [1]? [1]: stackoverflow.com/questions/7103064/…
Muhammad Hewedy
5
Si nous prenons en compte la suggestion de @Yaur de calculs entre fuseaux horaires, l'heure d'été devrait-elle affecter le calcul de quelque manière que ce soit?
DDM
6
Voté car il s'agit clairement d'une question de devoirs et aucune tentative existante n'a été fournie.
Marie

Réponses:

2123

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.Date > today.AddYears(-age)) age--;

Cependant, cela suppose que vous recherchez l' idée occidentale de l'âge et n'utilisez pas le calcul de l'Asie de l'Est .

Mike Polen
la source
252
Je voulais juste commenter les performances de DateTime.Now. Si vous n'avez pas besoin d'une valeur de fuseau horaire précise, utilisez DateTime.UtcNow, c'est beaucoup plus rapide.
JAG
104
Étant donné que nous parlons d'anniversaires, vous pouvez simplement utiliser DateTime.Today étant donné que la partie heure n'a pas de pertinence.
Tristan Warner-Smith
78
Cette réponse ne fonctionne pas avec tous les paramètres régionaux et tous les âges. Plusieurs pays ont sauté des dates après la naissance de personnes vivant actuellement, notamment la Russie (1918), la Grèce (1924) et la Turquie (1926).
Lars D
30
En fait, ce n'est pas encore tout à fait correct. Ce code suppose que «bday» est la partie date d'un DateTime. C'est un cas limite (je suppose que la plupart des gens passeront simplement des dates et non des dates-heures), mais si vous passez un anniversaire comme une date et une heure où l'heure est supérieure à 00:00:00, alors vous '' Je vais rencontrer le bug signalé par Danvil. La définition de bday = bday.Date résout ce problème.
Øyvind
119
La dernière ligne m'a fait trop réfléchir. Que diriez-vous plutôt: si (bday.AddYears (age)> now) age--; Cela semble être une expression plus intuitive.
cdiggins
1015

C'est une étrange façon de le faire, mais si vous formatez la date yyyymmddet soustrayez la date de naissance de la date actuelle, supprimez les 4 derniers chiffres que vous avez l'âge :)

Je ne connais pas C #, mais je crois que cela fonctionnera dans n'importe quel langage.

20080814 - 19800703 = 280111 

Déposez les 4 derniers chiffres = 28.

Code C #:

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

Ou bien sans toute la conversion de type sous la forme d'une méthode d'extension. Vérification d'erreur 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;
}
ScArcher2
la source
5
En fait, c'est idéal pour une utilisation sur MS-SQL avec des champs datetime (nombre total de jours depuis le 01-011900)
Patrik
5
@numerek Veuillez poster vos modifications suggérées comme leur propre réponse. Pour ce qu'elle vaut, l'année en cours multipliée par 10000 est loin d'être un débordement d'entier, de deux ordres de grandeur. 20,150,000 vs 2,147,483,648
GalacticCowboy
7
@LongChalk 20180101 - 20171231 = 8870. Déposez les 4 derniers chiffres et vous avez (un implicite) 0pour l'âge. Comment es-tu arrivé 1?
Rufus L
4
Je sais que c'est une vieille réponse mais je n'en ferais pas une méthode d'extension, ce n'est pas le bon endroit pour définir de telles logiques.
Lucca Ferri
1
Quel genre de sorcellerie est ce?
Muleskinner
391

Voici un extrait de test:

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

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;
}

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;
}
RMA
la source
33
Bien que ce code fonctionne, il affirme qu'une personne née un jour bissextile atteint l'année suivante le 1er mars les années non bissextiles, plutôt que le 28 février. En réalité, l'une ou l'autre option peut être correcte . Wikipédia a quelque chose à dire à ce sujet . Ainsi, même si votre code n'est pas "incorrect", la solution acceptée n'est pas non plus.
Matt Johnson-Pint
18
@MattJohnson Je pense que c'est en fait correct. Si mon anniversaire était le 29 février, alors le 28 février, mon anniversaire n'a pas passé et je devrais toujours avoir le même âge que le 27 février. Le 1er mars, cependant, nous avons dépassé mon anniversaire et je devrais avoir l'âge suivant. Aux États-Unis, une entreprise qui vend de l'alcool aura un signe qui dit quelque chose comme «Si vous êtes né après ce jour à YYYY, vous ne pouvez pas acheter d'alcool» (où YYYY change chaque année). Cela signifie qu'une personne née le 29 février ne peut pas acheter d'alcool le 28 février de l'année où elle atteint 21 ans (la plupart des endroits), et soutient l'idée qu'ils n'ont pas un an de plus avant le 1er mars.
jfren484
4
@ jfren484 - lisez l'article Wikipedia. Il varie considérablement d'une juridiction à l'autre.
Matt Johnson-Pint
9
@ jfren484 Votre revendication n'a absolument rien à voir avec la philosophie; mais tout à voir avec votre propre sentiment personnel . Lorsqu'une personne née le 29 février "vieillit" n'a pas d'importance à moins que l'âge ne forme une "limite d'âge légale" (par exemple, peut acheter de l'alcool, voter, obtenir une pension, rejoindre l'armée, obtenir un permis de conduire). Considérez l'âge américain pour boire (21 ans): pour la plupart des gens, c'est 7670 jours. Il est de 7671 jours s'il est né avant le 29 février de l'année bissextile ou à partir du 1er mars avant l'année bissextile. Si né le 29 février: le 28 février est de 7670 jours et le 1er mars est de 7671 jours. Le choix est arbitraire, il peut aller dans les deux sens.
Désillusionné
4
@CraigYoung Vous ne comprenez pas ce que je voulais dire par philosophie. J'ai utilisé ce terme par opposition à légalement. Si l'on écrit une demande qui doit connaître l'âge légal d'une personne, alors tout ce qu'il faut savoir, c'est comment les juridictions légales dans lesquelles sa demande est utilisée dans / pour traiter les personnes nées le 29 février. Si, cependant, nous sommes parler de la façon dont cela devrait être traité, alors c'est par définition de la philosophie. Et oui, l'opinion que j'ai donnée est ma propre opinion, mais comme je l'ai dit, je pense qu'il serait plus facile d'argumenter pour le 1er mars que pour le 28 février.
jfren484
110

La réponse simple à cela est d'appliquer AddYearscomme indiqué ci-dessous car 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évrier pour les années communes.

Certains estiment que le 1er mars est l'anniversaire des sauts mais ni .Net ni aucune règle officielle ne le soutiennent, et la logique commune n'explique pas pourquoi certains nés en février devraient avoir 75% de leurs anniversaires dans un autre mois.

De plus, une méthode Age se prête à être ajoutée comme extension à DateTime. Par cela, vous pouvez obtenir l'âge de la manière la plus simple possible:

  1. Élément de liste

int age = birthDate.Age ();

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 date critique est le suivant:

Date de naissance: 2000-02-29 Date ultérieure: 2011-02-28 Âge: 11

Production:

{
    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
}
camelCasus
la source
4
Un commentaire concernant l'anniversaire du 29 février le 1er mars, techniquement, l'avoir le 28 est trop tôt (1 jour plus tôt en fait). Le 1er est un jour trop tard. Mais comme l'anniversaire est entre, utiliser le 1er pour calculer l'âge en années non bissextiles me semble plus logique, car cette personne est en effet aussi âgée le 1er mars (et les 2 et 3) chaque année, mais pas le 28 février.
CyberClaw
1
Du point de vue de la conception de logiciels, l'écriture en tant que méthode d'extension n'a pas beaucoup de sens pour moi. date.Age(other)?
marsze
90

Ma suggestion

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

Il semble que l'année change à la bonne date. (J'ai repéré des tests jusqu'à 107 ans.)

James Curran
la source
26
Je ne pense pas que Harry Patch aurait apprécié votre méthodologie de tests ponctuels
MusiGenesis
3
Google ditdays in a year = 365.242199
mpen
12
La durée moyenne d'une année dans le calendrier grégorien est de 365,2425 jours.
dan04
4
Je dirais que c'est l'une des solutions les plus simples et c'est assez bon . Peu importe si je suis une demi-journée avant mon Xe anniversaire et le programme dit que j'ai X ans. Le programme est plus ou moins juste, mais pas mathématiquement. J'aime vraiment cette solution.
Peter Perháč
13
^^ Parce que parfois c'est important. Dans mes tests, cela échoue à l'anniversaire des personnes, il les signale plus jeunes qu'elles ne le sont.
ChadT
76

Une autre fonction, pas par moi mais trouvée sur le web et qui l'a affinée 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 des pays qui n'utilisent pas le calendrier grégorien? DateTime.Now est dans la culture spécifique au serveur, je pense. Je n'ai absolument aucune connaissance sur le fait de travailler avec des calendriers asiatiques et je ne sais pas s'il existe un moyen facile de convertir des dates entre calendriers, mais juste au cas où vous vous poseriez des questions sur ces chinois de l'année 4660 :-)

Michael Stum
la source
Cela semble gérer au mieux les différentes régions (formats de date).
webdad3
53

2 Les principaux problèmes à résoudre sont:

1. Calculez l'âge exact - en années, mois, jours, etc.

2. Calculer l'âge généralement perçu - les gens ne se soucient généralement pas de leur âge exact, ils se soucient seulement de leur anniversaire dans l'année en cours.


La solution pour 1 est évidente:

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

La solution pour 2 est celle qui n'est pas si précise pour déterminer l'âge total, mais qui est perçue comme précise par les gens. Les gens l'utilisent également généralement lorsqu'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
}

Notes à 2 .:

  • Ceci est ma solution préférée
  • Nous ne pouvons pas utiliser DateTime.DayOfYear ou TimeSpans, car ils décalent le nombre de jours dans les années bissextiles
  • J'ai mis un peu plus de lignes pour plus de lisibilité

Juste une note de plus ... Je créerais 2 méthodes de surcharge statique pour celui-ci, une pour une utilisation universelle, une seconde pour la convivialité:

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

public static int GetAge(DateTime birthDay) 
{ 
  return GetAge(birthDay, DateTime.Now);
}
Thetam
la source
50

Voici une ligne:

int age = new DateTime(DateTime.Now.Subtract(birthday).Ticks).Year-1;
SillyMonkey
la source
23
C'est cassé. Rendu testable: public static int CalculateAge (DateTime dateOfBirth, DateTime dateToCalculateAge) {return new DateTime (dateToCalculateAge.Subtract (dateOfBirth) .Ticks) .Year - 1; } ... Donne 14 ans lorsque j'entre le 1990-06-01 et calcule l'âge le jour AVANT son 14e anniversaire (1990-05-31).
Kjensen
43

Ceci est la version que nous utilisons ici. Cela 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 car cela sépare la logique de soustraction d'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 pouvez étendre l'opérateur ternaire pour le rendre encore plus clair, si vous pensez que ce genre de chose n'est pas clair.

Évidemment, cela se fait comme une méthode d'extension DateTime, mais vous pouvez clairement saisir cette seule ligne de code qui fait le travail et la placer n'importe où. Ici, nous avons une autre surcharge de la méthode Extension qui passe DateTime.Now, juste pour être complète.

David Wengier
la source
6
Je pense que cela peut être éteint d'un jour alors que exactement l'un de dateOfBirth ou dateAsAt tombe dans une année bissextile. Considérez l'âge d'une personne née le 1er mars 2003 le 29 février 2004. Pour rectifier cela, vous devez faire une comparaison lexicographique des paires (Month, DayOfMonth) et l'utiliser pour le conditionnel.
Doug McClean
1
ça ne va pas non plus montrer le bon âge à partir de votre anniversaire.
dotjoe
43

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);
Nick Berardi
la source
34

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;
    }
}
Elmer
la source
32

Cela donne "plus de détails" à cette question. C'est peut-être 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);
Jacqueline Loriault
la source
1
Cela ne fonctionne pas tout le temps. L'ajout d'une étendue à DateTime.MinValue pourrait fonctionner boes, cela ne tient pas compte des années bissextiles, etc. .Maintenant.
Athanasios Kataras
3
la durée elle-même prend automatiquement en compte les années bissextiles entre 2 dates, donc je ne suis pas sûr de ce que vous faites. J'ai posé des questions sur les forums Microsoft et Microsoft a confirmé qu'il prend en compte les années bissextiles entre 2 dates.
Jacqueline Loriault
2
Considérez les DEUX senarios suivants. 1st DateTime.Now est 1/1/2001 et un enfant est né le 1/1/2000. 2000 est une année bissextile et le résultat sera de 1 an, 0 mois et 1 jour. Dans le deuxième senarion, DateTime est maintenant le 1/1/2002 et l'enfant est né le 1/1/2001. Dans ce cas, le résultat sera de 1 an, 0 mois et 0 jour. Cela se produira parce que vous ajoutez la durée sur une année non bissextile. Si DateTime.MinValue était une année bissextile, les résultats seraient de 1 an au premier et de 0 an, 11 mois et 30 jours. (Essayez-le dans votre code).
Athanasios Kataras
1
Upvote! J'ai trouvé une solution à peu près identique (j'ai utilisé DateTime.MinValue.AddTicks (span.Ticks) au lieu de +, mais le résultat est le même et le vôtre a quelques caractères de moins).
Makotosan
4
Vous avez tout à fait raison, ce n'est pas le cas. Mais SI c'était ce qui serait le résultat. En quoi est-ce important? Ce n'est pas le cas. Dans les deux cas, saut ou pas, il existe des exemples où cela ne fonctionne pas. C'est ce que je voulais montrer. Le DIFF est correct. La durée prend en compte les années bissextiles. Mais AJOUTER à une date de base ne l'est pas. Essayez les exemples dans le code et vous verrez que j'ai raison.
Athanasios Kataras
28

J'ai créé une fonction définie par l'utilisateur SQL Server pour calculer l'âge d'une personne, compte tenu de 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;
    }
};
2601
la source
28

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;
}

Cela a été largement testé à l'unité. Ça a l'air 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ée d'ici ) est:

Fixons 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, c'est Yn-Ybsi la date a déjà été atteinte, Yn-Yb-1sinon.

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

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

Avec division entière

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

b) Si Mn=Mbet Dn<Db, nous avons31*(Mn - Mb) = 0 and -30 <= Dn-Db <= -1

Avec une division entière, encore une fois

(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 division entière

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

d) Si Mn=Mbet Dn>Db, nous avons 31*(Mn - Mb) = 0 and 1 <= Dn-Db <= 30

Avec une division entière, encore une fois

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

e) Si Mn=Mbet Dn=Db, nous avons31*(Mn - Mb) + Dn-Db = 0

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

Matthew Watson
la source
3
Je suis tombé sur cette longue et ennuyeuse discussion, et votre solution est une approche vraiment sympa et petite. Merci de rester simple
nabuchodonossor
25

J'ai passé un peu de temps à travailler là-dessus et je l'ai trouvé pour calculer l'âge d'une personne 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++;
    }

}
Jon
la source
21

Faut-il considérer les personnes de moins d'un an? en tant que culture chinoise, nous décrivons l'âge des petits bébés comme 2 mois ou 4 semaines.

Ci-dessous est mon implémentation, ce n'est pas aussi simple que ce que j'imaginais, surtout pour faire face à des dates comme le 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 est passée sous les cas de test.

[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.

rockXrock
la source
20

Garder les choses simples (et peut-être stupides :)).

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.");
181261
la source
TimeSpan était mon premier choix, mais a constaté qu'il ne propose pas de propriété TotalYears. Vous pouvez essayer (ts.TotalDays / 365) - mais cela ne tient pas compte des années bissextiles, etc.
Lazlow
19

La façon la plus simple que j'ai jamais trouvée est la suivante. Il fonctionne correctement pour les États-Unis et les régions d'Europe occidentale. Je ne peux pas parler à d'autres endroits, en particulier des endroits comme la Chine. 4 comparaisons supplémentaires, 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 cherchais les réponses à cette question et j'ai remarqué que personne n'a fait référence aux implications réglementaires / juridiques des naissances bissextiles. Par exemple, selon Wikipédia , si vous êtes né le 29 février dans différentes juridictions, votre anniversaire de non-année bissextile 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 aux fins du permis de conduire et le 1er mars à d'autres fins.
  • Taïwan: c'est le 28 février.

Et d'après ce que je peux dire, 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.

A cet effet, 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 convient de noter que ce code suppose:

  • Un calcul occidental (européen) de l'âge, et
  • Un calendrier, comme le calendrier grégorien qui insère un seul jour bissextile à la fin d'un mois.
Nicholas Carey
la source
19
TimeSpan diff = DateTime.Now - birthdayDateTime;
string age = String.Format("{0:%y} years, {0:%M} months, {0:%d}, days old", diff);

Je ne sais pas exactement comment vous souhaitez qu'il vous soit retourné, alors je viens de créer une chaîne lisible.

Dakotah Hicock
la source
18

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 précise 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, ergo, la bonne réponse générique devrait être (bien sûr en supposant une normalisation DateTimeet sans tenir compte des 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--;

En finance, il y a un problème similaire lors du calcul de quelque chose souvent appelé la fraction du nombre de jours , qui correspond approximativement à un certain 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 la convention réelle / réelle (en comptant tous les jours "correctement"):

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 type qui a nommé cette convention de date doit sérieusement avoir fait un trippin):

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

Je me demande combien de temps il nous reste à faire avant qu'un âge relativiste en secondes ne devienne plus utile que l'approximation approximative des cycles de la terre autour du soleil jusqu'à présent :) Ou en d'autres termes, quand une période doit avoir un emplacement ou une fonction représentant le mouvement pour être valide :)

flindeberg
la source
Qu'est-ce que TickFactor ?
Protiguous il y a
@ Ticks protégés par seconde, utilisés pour normaliser les ticks en secondes.
flindeberg il y a
17

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);
Rajeshwaran SP
la source
Avec un enchaînement de cordes, cela serait possible: 47 ans 11 Mo 7 jours
JoshYates1980
16

C'est l'une des réponses les plus précises qui soit en mesure de résoudre l'anniversaire du 29 février par rapport à 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;
}



mjb
la source
C'est aujourd'hui! (Le prochain est dans quatre ans.)
Peter Mortensen
15

J'ai une méthode personnalisée pour calculer l'âge, plus un message de validation de bonus au cas où cela aiderait:

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
}

Appelez ici la méthode et transmettez la valeur datetime (MM / jj / aaaa si le serveur est défini sur les paramètres régionaux des États-Unis). Remplacez-le par n'importe quoi une boîte de message ou n'importe quel conteneur à 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

N'oubliez pas que vous pouvez formater le message comme vous le souhaitez.

Knickerless-Noggins
la source
14

Et 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"; ;
}
Doron
la source
12
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;
}
AEMLoviji
la source
10

L'approche suivante (extrait de Time Period Library pour la classe .NET DateDiff ) prend en compte le calendrier des informations de culture:

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

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

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

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

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

Usage:

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

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

Cette question classique mérite une solution Noda Time .

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

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

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

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

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

    return (int) period.Years;
}

Usage:

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

Vous pourriez également être intéressé par les améliorations suivantes:

  • Passer dans l'horloge en tant que IClock, au lieu d'utiliser SystemClock.Instance, améliorerait la testabilité.

  • Le fuseau horaire cible changera probablement, vous voudrez donc également un DateTimeZoneparamètre.

Voir aussi mon billet de blog sur ce sujet: Gestion des anniversaires et autres anniversaires

Matt Johnson
la source
Êtes-vous affilié à Noda Time?
Zimano
J'y ai contribué, mais c'est principalement Jon Skeet.
Matt Johnson-Pint
9

J'ai utilisé la solution de ScArcher2 pour un calcul précis de l'année de l'âge d'une personne, mais je devais aller plus loin et calculer leurs mois et jours ainsi que 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;
}
Dylan Hayes
la source
9

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  
xenedia
la source
8

J'ai apporté un petit changement à la réponse de Mark Soen : j'ai réécrit la troisième ligne afin que l'expression puisse ê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;
}

Je l'ai également transformé en fonction pour plus de clarté.

cdiggins
la source