Как рассчитать возраст человека на основе дня рождения типа DateTime?

Учитывая DateTime, представляющий день рождения человека, как мне рассчитать его возраст в годах?

что до сих пор упускали из виду все ответы, так это то, что это зависит от того, где человек родился и где он сейчас находится.

Yaur 21.05.2011 11:34

@Yaur: Просто преобразуйте текущее время + время рождения в GMT / UTC, возраст - это только относительное значение, поэтому часовые пояса не имеют значения. Для определения текущего часового пояса пользователя вы можете использовать GeoLocating.

Stefan Steiger 03.10.2011 14:20

Почему бы не рассмотреть [Юлианскую дату] [1]? [1]: stackoverflow.com/questions/7103064/…

Muhammad Hewedy 05.10.2013 17:32

Если мы принимаем во внимание предложение @Yaur о расчетах между часовыми поясами, должно ли летнее время влиять на расчет каким-либо образом?

DDM 11.07.2015 06:42

Обратите внимание, что для лиц младше одного года их возраст указывается в днях, неделях или месяцах. Время перехода для блоков может зависеть от домена.

Andrew Morton 11.11.2017 01:09

Как мы все видим, окончательного определения возраста не существует. Многие женщины, которых я встречал, имеют тенденцию округлять свое время жизни до целого года до двадцати с небольшим лет, а затем начинают округлять в меньшую сторону. Я родился 3 января, поэтому я просто вычитаю текущий год из года моего рождения, независимо от того, какой он день. некоторые люди думают, что если вы родились в високосный день, вы стареете в соотношении 1/4. Что, если бы вы родились в високосную секунду? считается ли 8-месячный ребенок за 1? Если я лечу на запад, стану ли я моложе? Если мое сердце остановится на минуту, следует ли мне включить это в расчет?

Erdogan Kurtur 15.10.2020 19:27

Сценарий, который не учитывается всеми ответами: мой прадедушка родился 29 февраля 1928 года. Он умер в позапрошлом году, всего за 13 месяцев до своего 23-го дня рождения.

Rab 14.01.2021 20:03

@Rab - вообще говоря, возраст человека, родившегося в високосный год, не рассчитывается таким образом .... но грамматика выше потребовала времени, чтобы выяснить.

JosephDoggie 17.02.2021 19:56

Это второй из существующих общедоступных вопросов!

Anonymous 12.03.2021 03:22
Стоит ли изучать PHP в 2026-2027 годах?
Стоит ли изучать PHP в 2026-2027 годах?
Привет всем, сегодня я хочу высказать свои соображения по поводу вопроса, который я уже много раз получал в своем сообществе: "Стоит ли изучать PHP в...
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
В JavaScript одним из самых запутанных понятий является поведение ключевого слова "this" в стрелочной и обычной функциях.
Приемы CSS-макетирования - floats и Flexbox
Приемы CSS-макетирования - floats и Flexbox
Здравствуйте, друзья-студенты! Готовы совершенствовать свои навыки веб-дизайна? Сегодня в нашем путешествии мы рассмотрим приемы CSS-верстки - в...
Тестирование функциональных ngrx-эффектов в Angular 16 с помощью Jest
В системе управления состояниями ngrx, совместимой с Angular 16, появились функциональные эффекты. Это здорово и делает код определенно легче для...
Концепция локализации и ее применение в приложениях React ⚡️
Концепция локализации и ее применение в приложениях React ⚡️
Локализация - это процесс адаптации приложения к различным языкам и культурным требованиям. Это позволяет пользователям получить опыт, соответствующий...
Пользовательский скаляр GraphQL
Пользовательский скаляр GraphQL
Листовые узлы системы типов GraphQL называются скалярами. Достигнув скалярного типа, невозможно спуститься дальше по иерархии типов. Скалярный тип...
1 982
9
660 115
64
Перейти к ответу Данный вопрос помечен как решенный

Ответы 64

Лучший способ, который я знаю из-за високосных лет и всего остального:

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

Еще одна функция, не мной, но найденная в Интернете и немного доработанная:

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

Мне приходят в голову только две вещи: а как насчет людей из стран, которые не используют григорианский календарь? Я думаю, DateTime.Now находится в серверной культуре. У меня абсолютно нулевой опыт работы с азиатскими календарями, и я не знаю, есть ли простой способ конвертировать даты между календарями, но на всякий случай вам интересно узнать об этих китайских парнях из 4660 года :-)

Похоже, что это лучше всего подходит для разных регионов (форматов даты).

webdad3 10.11.2016 01:06
Ответ принят как подходящий

Легкое для понимания и простое решение.

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

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

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

Однако это предполагает, что вы ищете информацию о возрасте западный, а не используете Восточноазиатский расчет.

Этот ответ не работает для всех регионов и всех возрастов. Некоторые страны пропустили даты после рождения ныне живущих людей, в том числе Россия (1918 г.), Греция (1924 г.) и Турция (1926 г.).

Lars D 10.11.2009 01:09

Собственно, это еще не совсем так. Этот код предполагает, что bday - это часть даты DateTime. Это крайний случай (я предполагаю, что большинство людей просто будут передавать даты, а не дату-время), но если вы передадите день рождения как дату и время, где время больше 00:00:00, тогда вы ' Я столкнусь с ошибкой, на которую указал Данвил. Установка bday = bday.Date исправляет это.

Øyvind 16.11.2010 18:37

Это та версия, которую мы здесь используем. Это работает, и это довольно просто. Это та же идея, что и у Джеффа, но я думаю, что она немного яснее, потому что разделяет логику вычитания единицы, поэтому ее немного легче понять.

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

Вы можете расширить тернарный оператор, чтобы сделать его еще более ясным, если считаете, что что-то в этом роде непонятно.

Очевидно, что это делается как метод расширения на DateTime, но очевидно, что вы можете взять одну строчку кода, которая выполняет эту работу, и поместить ее куда угодно. Здесь у нас есть еще одна перегрузка метода Extension, который передается в DateTime.Now, просто для полноты.

Я думаю, что это может быть отключено на один день, когда в високосный год выпадает ровно одно из dateOfBirth или dateAsAt. Рассмотрим возраст человека, родившегося 1 марта 2003 г. и 29 февраля 2004 г. Чтобы исправить это, вам необходимо провести лексикографическое сравнение пар (Месяц, День-месяц) и использовать его для условного выражения.

Doug McClean 23.12.2008 18:36

он также не будет показывать правильный возраст вашего дня рождения.

dotjoe 30.01.2009 00:19

Это странный способ сделать это, но если вы отформатируете дату до yyyymmdd и вычтите дату рождения из текущей даты, тогда отбросьте последние 4 цифры вашего возраста :)

Я не знаю C#, но верю, что это будет работать на любом языке.

20080814 - 19800703 = 280111 

Отбросьте последние 4 цифры = 28.

Код C#:

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

Или, в качестве альтернативы, без преобразования всех типов в виде метода расширения. Проверка ошибок опущена:

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

На самом деле это отлично подходит для использования в MS-SQL с полями datetime (общее количество дней с 01-011900)

Patrik 03.07.2015 15:01

в альтернативном ответе вы можете избежать целочисленного переполнения, вычтя годы, затем вычтите месяц * 30,5 + день и разделите на 366

numerek 03.09.2015 23:14

@numerek Пожалуйста, опубликуйте свои предлагаемые изменения в качестве собственного ответа. Как бы то ни было, текущий год, умноженный на 10000, далек от целочисленного переполнения на два порядка. 20,150,000 и 2,147,483,648

GalacticCowboy 03.09.2015 23:23

Было бы лучше добавить параметр: float.Parse (x, CultureInfo.InvariantCulture), потому что некоторые местные жители используют запятую вместо точки для разделителя в дробных числах.

Vlad Krylov 28.03.2016 11:39

Год DateTime.MaxValue равен 9999, год MinValue равен 1, переполнение невозможно.

Antonín Lejsek 01.01.2017 05:07

В этом ответе предполагается, что у детей високосного дня свои дни рождения 1 марта невисокосных лет.

Jamie Kitson 13.02.2018 12:38

задаю вопрос по очень старому посту int age = (now - dob) / 10000;, почему вы вычитаете на 10000?

Monojit Sarkar 25.02.2018 11:24

@LongChalk 20180101 - 20171231 = 8870. Отбросьте последние 4 цифры, и вы получите (подразумеваемый) 0 для возраста. Как вы получили 1?

Rufus L 14.06.2018 23:36

@RufusL Это 0, а не 1. floor(8870 / 10000) == 0. Вы «считаете» десять тысяч, а в 8870 у вас ноль десять тысяч.

flindeberg 03.07.2018 21:13

Приступить к этому довольно поздно, но вы также можете включить проверку различий TimeOfDay, чтобы получить точный возраст, по крайней мере, на C#.

JansthcirlU 26.11.2020 20:31

Я создал функцию, определяемую пользователем SQL Server, чтобы вычислить возраст человека с учетом даты его рождения. Это полезно, когда вам это нужно как часть запроса:

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

Я думаю, что в TimeSpan есть все, что нам нужно, без необходимости прибегать к 365,25 (или любому другому приближению). Расширяя пример Августа:

DateTime myBD = new DateTime(1980, 10, 10);
TimeSpan difference = DateTime.Now.Subtract(myBD);

textBox1.Text = difference.Years + " years " + difference.Months + " Months " + difference.Days + " days";

Неа. TimeSpan как дни, но не месяцы или годы

James Curran 04.10.2008 00:13

Мое предложение

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

Кажется, что год меняется в нужную дату. (Я прошел тестирование до 107 лет.)

Не думаю, что Гарри Патч оценил бы вашу методологию выборочного тестирования: latimes.com/news/obituaries/…

MusiGenesis 01.08.2009 20:03

Google говорит days in a year = 365.242199

mpen 12.08.2010 09:28

Средняя продолжительность года по григорианскому календарю составляет 365,2425 дней.

dan04 06.10.2010 06:01

Я бы сказал, это одно из самых простых решений и это достаточно хорошо. Какая разница, если я за полдня до своего X дня рождения, а программа говорит, что мне X лет. Программа более-менее правильная, хотя и не математически. Мне очень нравится это решение.

Peter Perháč 09.03.2011 15:07

^^ Потому что иногда это важно. В моем тестировании это не удается в день рождения человека, он сообщает о том, что он моложе, чем он.

ChadT 25.03.2011 08:27

Я бы тоже начал с этого, но при тестировании он часто ошибается с датой рождения человека.

Dave Cousineau 01.01.2016 23:51

@ChadT, какое решение вы придумали, чтобы оно работало? пользователи постоянно жалуются на то, что День рождения не работает в их дни рождения, и смотрят, как истекают часы.

myusuf3 07.11.2016 20:55

Не собираюсь тестировать его, но может показаться, что если у вас проблема только в том, что вы хотите, чтобы результат увеличивался на день раньше, вам, вероятно, удастся обойтись: int age = (int) (((DateTime.Now - bday).TotalDays+1)/365.242199);

krowe2 21.11.2018 18:44

Его следует проводить выборочные испытания в течение четырех лет подряд, поскольку ошибка накапливается (сбрасывается на один день каждые четыре года к високосному дню - как сегодня!).

Peter Mortensen 29.02.2020 04:26

Я потратил некоторое время на работу над этим и придумал это, чтобы вычислить чей-то возраст в годах, месяцах и днях. Я тестировал проблему 29 февраля и високосные годы, и, похоже, это сработало, буду признателен за любые отзывы:

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

}

Вот однострочник:

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

Сломано. Сделано тестируемым: общедоступный статический int CalculateAge (DateTime dateOfBirth, DateTime dateToCalculateAge) {return new DateTime (dateToCalculateAge.Subtract (dateOfBirth) .Ticks) .Yea‌ r - 1; } ... Дает возраст 14, когда я ввожу 1990-06-01 и вычисляю возраст на день ДО его 14-летия (1990-05-31).

Kjensen 06.02.2011 00:42

Вот решение.

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

Со строкой concat это было бы возможно: 47 лет 11 пн 7 дней

JoshYates1980 14.06.2018 18:58

Вот фрагмент теста:

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

Вот вам методы:

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

Хотя этот код работает, он утверждает, что человек, родившийся в високосный день, достигает следующего года возраста 1 марта в невисокосные годы, а не 28 февраля. На самом деле любой вариант может быть правильным. Википедия может что-то сказать об этом. Так что, хотя ваш код не «неправильный», ни одно решение не является общепринятым.

Matt Johnson-Pint 17.08.2014 09:44

@MattJohnson Я думаю, что это действительно так. Если мой день рождения был 29 февраля, то 28 февраля мой день рождения не прошел, и я все еще должен быть в том же возрасте, что и 27 февраля. Однако 1 марта мой день рождения закончился, и я должен достичь следующего возраста. В США у предприятия, продающего алкоголь, будет вывеска, на которой написано что-то вроде «Если вы родились после этого дня в ГГГГ, вы не можете покупать алкоголь» (где ГГГГ меняется каждый год). Это означает, что человек, родившийся 29 февраля, не может покупать алкоголь 28 февраля того года, когда ему исполняется 21 год (в большинстве случаев), и поддерживает идею о том, что они не на год старше до 1 марта.

jfren484 12.07.2016 20:18

@ jfren484 - прочтите статью в Википедии. Он значительно варьируется в зависимости от юрисдикции.

Matt Johnson-Pint 12.07.2016 22:26

@ jfren484 Ваше утверждение не имеет ничего общего с философией; но все дело в ваше собственное личное чувство. Когда человек, родившийся 29 февраля, «возраст» не имеет большого значения, если только возраст не образует «возрастную границу» (например, может покупать алкоголь, голосовать, получать пенсию, служить в армии, получать водительские права). Рассмотрим возраст употребления алкоголя в США (21 год): для большинства людей это 7670 дней. Это 7671 день, если они родились до 29 февраля в високосном году или с 1 марта до високосного года. При рождении 29 февраля: 28 февраля - 7670 дней, а 1 марта - 7671 день. Выбор произвольный может идти в любом направлении.

Disillusioned 04.03.2017 13:06

@CraigYoung Вы не понимаете, что я имел в виду с философской точки зрения. Я использовал этот термин как противопоставление юридически. Если кто-то пишет приложение, в котором необходимо знать возраст человека, то все, что ему нужно знать, - это то, как в правовых юрисдикциях, в которых его приложение используется в / для лечения людей, родившихся 29 февраля. Если, однако, мы если говорить о том, как обращаться с этим должен, то это по определению философия. И да, мнение, которое я высказал, является моим собственным мнением, но, как я уже сказал, я думаю, что было бы легче выступать за 1 марта, чем за 28 февраля.

jfren484 04.03.2017 21:45

Это может сработать:

public override bool IsValid(DateTime value)
{
    _dateOfBirth =  value;
    var yearsOld = (double) (DateTime.Now.Subtract(_dateOfBirth).TotalDays/365);
    if (yearsOld > 18)
        return true;
    return false;
}

Вау. Почему значение является объектом, а не DateTime? Сигнатура метода должна быть public override bool Is18OrOlder(DateTime birthday). А как быть с людьми, родившимися 29 февраля? Кто сказал, что мы пытаемся проверить, исполнилось ли пользователю 18 лет? Вопрос был в том, "как я могу рассчитать чей-то возраст?"

Chris Shouts 05.05.2010 00:57

Как это случилось? Я даже не помню, как ставил IsValid как объект. Это должно быть DateTime!

azamsharp 05.05.2010 04:03

Почему бы не использовать if вместо заявления return yearsOld > 18;?

T_Bacon 28.03.2018 11:41

Я создал структуру Age, которая выглядит так:

public struct Age : IEquatable<Age>, IComparable<Age>
{
    private readonly int _years;
    private readonly int _months;
    private readonly int _days;

    public int Years  { get { return _years; } }
    public int Months { get { return _months; } }
    public int Days { get { return _days; } }

    public Age( int years, int months, int days ) : this()
    {
        _years = years;
        _months = months;
        _days = days;
    }

    public static Age CalculateAge( DateTime dateOfBirth, DateTime date )
    {
        // Here is some logic that ressembles Mike's solution, although it
        // also takes into account months & days.
        // Ommitted for brevity.
        return new Age (years, months, days);
    }

    // Ommited Equality, Comparable, GetHashCode, functionality for brevity.
}

Вот небольшой пример кода для C#, который я придумал, будьте осторожны с крайними случаями, особенно високосными годами, не все вышеперечисленные решения учитывают их. Выдвижение ответа как DateTime может вызвать проблемы, поскольку вы можете в конечном итоге попытаться поместить слишком много дней в конкретный месяц, например. 30 дней в февр.

public string 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 = 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++;
    }

    return "Years: " + years + " Months: " + months + " Days: " + days;
}

Мне больше всего нравится это решение, однако при подсчете месяцев оно должно быть if (months> = 12). Попробуйте 6-8-2012 - 6-4-1993 для проверки.

Jerry 08.06.2012 23:43

Я использую это:

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

Делать это просто (и, возможно, глупо :)).

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 был моим первым выбором, но обнаружил, что он не предлагает свойство TotalYears. Вы можете попробовать (ts.TotalDays / 365), но он не учитывает високосные годы и т. д.

Lazlow 22.09.2011 00:14

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

Самый простой способ, который я когда-либо находил, - это. Он правильно работает для регионов США и Западной Европы. Не могу разговаривать с другими местами, особенно с такими местами, как Китай. Не более 4 дополнительных сравнений после первоначального вычисления возраста.

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

Я просматривал ответы на этот вопрос и заметил, что никто не упомянул нормативные / правовые последствия високосных родов. Например, согласно Википедии, если вы родились 29 февраля в разных юрисдикциях, ваш день рождения не в високосный год может быть разным:

  • В Великобритании и Гонконге: это порядковый день в году, поэтому на следующий день, 1 марта, ваш день рождения.
  • В Новой Зеландии: это предыдущий день, 28 февраля для целей выдачи водительских прав и 1 марта для других целей.
  • Тайвань: 28 февраля.

И, насколько я могу судить, в США законодательные акты по этому поводу ничего не говорят, оставляя это на усмотрение общего права и того, как различные регулирующие органы определяют вещи в своих правилах.

С этой целью улучшение:

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

Следует отметить, что этот код предполагает:

  • Западный (европейский) расчет возраста, и
  • Календарь, подобный григорианскому календарю, в котором в конце месяца добавляется один високосный день.

Простой ответ на этот вопрос - применить AddYears, как показано ниже, потому что это единственный собственный метод добавления лет к 29 февраля високосных лет и получения правильного результата 28 февраля для обычных лет.

Некоторые считают, что 1 марта - день рождения прыгунов, но ни .Net, ни какие-либо официальные правила не поддерживают этого, и не существует общепринятой логики, объясняющей, почему у некоторых рожденных в феврале 75% дней рождения должны быть в другом месяце.

Кроме того, метод Age может быть добавлен как расширение DateTime. Таким образом можно получить возраст самым простым способом:

  1. Элемент списка

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

Теперь запустим этот тест:

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

Вот пример критической даты:

Дата рождения: 29.02.2000 Позднее: 28.02.2011 Возраст: 11

Выход:

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

И для более поздней даты 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
}

Комментарий относительно того, что 29 февраля день рождения должен быть 1 марта, технически, иметь его 28-го слишком рано (фактически, на 1 день раньше). 1-го уже на один день поздно. Но поскольку день рождения находится между ними, использование 1-го числа для расчета возраста в невисокосных годах для меня имеет больше смысла, поскольку этот человек действительно становится таким же старым 1 (и 2 и 3) марта каждого года, но не 28 февраля.

CyberClaw 24.07.2018 19:31

С точки зрения разработки программного обеспечения, писать это как метод расширения для меня не имеет большого смысла. date.Age(other)?

marsze 12.12.2018 11:11

Как насчет этого решения?

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

2 Основные проблемы, которые необходимо решить:

1. Рассчитайте точный возраст - в годах, месяцах, днях и т. д.

2. Рассчитайте общепринятый возраст. - людям обычно все равно, сколько им лет, им просто важно, когда у них день рождения в текущем году.


Решение для 1 очевидно:

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

Решение для 2 - это решение, которое не так точно определяет общий возраст, но воспринимается людьми как точное. Люди также обычно используют его, когда вычисляют свой возраст «вручную»:

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
}

Примечания к 2 .:

  • Это мое предпочтительное решение
  • Мы не можем использовать DateTime.DayOfYear или TimeSpans, поскольку они меняют количество дней в високосных годах.
  • Я добавил немного больше строк для удобочитаемости

Еще одно замечание ... Я бы создал для него 2 статических перегруженных метода, один для универсального использования, второй для удобства использования:

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

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

Следующий подход (выдержка из Библиотека периодов времени для .NET класса DateDiff) рассматривает календарь информации о культуре:

// ----------------------------------------------------------------------
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

Использование:

// ----------------------------------------------------------------------
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

Вот расширитель DateTime, который добавляет вычисление возраста к объекту DateTime.

public static class AgeExtender
{
    public static int GetAge(this DateTime dt)
    {
        int d = int.Parse(dt.ToString("yyyyMMdd"));
        int t = int.Parse(DateTime.Today.ToString("yyyyMMdd"));
        return (t-d)/10000;
    }
}

тьфу, не делай этого. ToString и int.Parse относительно дороги, и, хотя я против микрооптимизации, скрывать дорогостоящие функции в методах расширения, которые должны быть тривиальными операциями, не является хорошей идеей.

Yaur 21.05.2011 11:31

Кроме того, это дубликат ответа ScArcher2: stackoverflow.com/questions/9/…

David Schmitt 30.05.2011 17:32

Яур, мне очень нравится решение Элмера, основанное на DayOfYear, вероятно, более эффективное, чем мое. Обратите внимание, что моей целью не было изменить алгоритм ScArcher2, я чувствовал, что это было бы грубо. Это было просто, чтобы показать, как реализовать метод расширения.

B2K 09.09.2011 02:23

Я сделал одно небольшое изменение в ответе Марка Соена: я переписал третью строку, чтобы выражение можно было легче анализировать.

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

Я также превратил его в функцию для ясности.

Я использовал решение ScArcher2 для точного расчета возраста людей, но мне нужно было пойти дальше и вычислить их месяцы и дни вместе с годами.

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

Я хочу добавить вычисления календаря на иврите (или другой календарь System.Globalization можно использовать таким же образом), используя переписанные функции из этого потока:

Public Shared Function CalculateAge(BirthDate As DateTime) As Integer
    Dim HebCal As New System.Globalization.HebrewCalendar ()
    Dim now = DateTime.Now()
    Dim iAge = HebCal.GetYear(now) - HebCal.GetYear(BirthDate)
    Dim iNowMonth = HebCal.GetMonth(now), iBirthMonth = HebCal.GetMonth(BirthDate)
    If iNowMonth < iBirthMonth Or (iNowMonth = iBirthMonth AndAlso HebCal.GetDayOfMonth(now) < HebCal.GetDayOfMonth(BirthDate)) Then iAge -= 1
    Return iAge
End Function

Вопрос: «Как в C# рассчитать чей-то возраст ...».

Protiguous 05.06.2020 13:20

C# и vb одинаковы для этой идеи.

Moshe L 06.06.2020 20:32

Это просто и кажется мне точным. Я делаю предположение в целях високосных лет, что независимо от того, когда человек решает отметить день рождения, технически он не старше на год, пока с его последнего дня рождения не пройдет 365 дней (т.е. 28 февраля не делает его на год старше) .

DateTime now = DateTime.Today;
DateTime birthday = new DateTime(1991, 02, 03);//3rd feb

int age = now.Year - birthday.Year;

if (now.Month < birthday.Month || (now.Month == birthday.Month && now.Day < birthday.Day))//not had bday this year yet
  age--;

return age;

Попробуйте это решение, оно работает.

int age = (Int32.Parse(DateTime.Today.ToString("yyyyMMdd")) - 
           Int32.Parse(birthday.ToString("yyyyMMdd rawrrr"))) / 10000;

Это не прямой ответ, а скорее философское рассуждение о рассматриваемой проблеме с квазинаучной точки зрения.

Я бы сказал, что вопрос не определяет единицу измерения или культуру, в которой измеряется возраст, большинство ответов, похоже, предполагают целочисленное годовое представление. Единицей СИ для времени является second, следовательно, правильный общий ответ должен быть (конечно, принимая нормализованный DateTime и не принимая во внимание релятивистские эффекты):

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

Согласно христианскому способу исчисления возраста в годах:

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

В финансах есть аналогичная проблема при вычислении чего-то, что часто называют Доля подсчета дней, что примерно соответствует количеству лет для данного периода. И проблема возраста - это действительно проблема измерения времени.

Пример условного обозначения фактического / фактического (считая все дни "правильно"):

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;

Другой довольно распространенный способ измерения времени - "сериализация" (чувак, назвавший это соглашение о дате, должно быть, серьезно споткнулся):

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

Интересно, сколько времени нам нужно пройти до того, как релятивистская эпоха в секундах станет более полезной, чем грубое приближение циклов Земля вокруг Солнца в течение всей жизни человека :) Или, другими словами, когда период должен быть определен местом или функция, представляющая движение, чтобы быть действительной :)

Что такое TickFactor?

Protiguous 05.06.2020 13:22

@Protiguous Ticks в секунду, используется для нормализации тиков до секунд.

flindeberg 05.06.2020 17:52

Нужно ли учитывать людей младше 1 года? В китайской культуре мы описываем возраст маленьких детей как 2 месяца или 4 недели.

Ниже представлена ​​моя реализация, она не так проста, как я себе представлял, особенно с датой вроде 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";
    }
}

Эта реализация прошла тестовые примеры ниже.

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

Надеюсь, это поможет.

Вот очень простой и понятный пример.

private int CalculateAge()
{
//get birthdate
   DateTime dtBirth = Convert.ToDateTime(BirthDatePicker.Value);
   int byear = dtBirth.Year;
   int bmonth = dtBirth.Month;
   int bday = dtBirth.Day;
   DateTime dtToday = DateTime.Now;
   int tYear = dtToday.Year;
   int tmonth = dtToday.Month;
   int tday = dtToday.Day;
   int age = tYear - byear;
   if (bmonth < tmonth)
       age--;
   else if (bmonth == tmonth && bday>tday)
   {
       age--;
   }
return age;
}

Вот еще один ответ:

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

Это было тщательно протестировано. Это действительно выглядит немного "волшебно". Число 372 - это количество дней в году, если бы каждый месяц состоял из 31 дня.

Объяснение того, почему это работает (снят отсюда):

Let's set 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

We know that what we need is either Yn-Yb if the date has already been reached, Yn-Yb-1 if it has not.

a) If Mn<Mb, we have -341 <= 31*(Mn-Mb) <= -31 and -30 <= Dn-Db <= 30

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

With integer division

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

b) If Mn=Mb and Dn<Db, we have 31*(Mn - Mb) = 0 and -30 <= Dn-Db <= -1

With integer division, again

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

c) If Mn>Mb, we have 31 <= 31*(Mn-Mb) <= 341 and -30 <= Dn-Db <= 30

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

With integer division

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

d) If Mn=Mb and Dn>Db, we have 31*(Mn - Mb) = 0 and 1 <= Dn-Db <= 30

With integer division, again

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

e) If Mn=Mb and Dn=Db, we have 31*(Mn - Mb) + Dn-Db = 0

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

С меньшим количеством конверсий и UtcNow этот код может позаботиться о людях, родившихся 29 февраля високосного года:

public int GetAge(DateTime DateOfBirth)
{
    var Now = DateTime.UtcNow;
    return Now.Year - DateOfBirth.Year -
        (
            (
                Now.Month > DateOfBirth.Month ||
                (Now.Month == DateOfBirth.Month && Now.Day >= DateOfBirth.Day)
            ) ? 0 : 1
        );
}

Это сегодня! (Следующий - через четыре года.)

Peter Mortensen 29.02.2020 04:11

How come the MSDN help did not tell you that? It looks so obvious:

System.DateTime birthTime = AskTheUser(myUser); // :-)
System.DateTime now = System.DateTime.Now;
System.TimeSpan age = now - birthTime; // As simple as that
double ageInDays = age.TotalDays; // Will you convert to whatever you want yourself?

Что ж, это фантастика, если вы находитесь в одной из нулевых стран на Земле, где возраст взрослых людей измеряется днями.

Casey 15.10.2014 00:47

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

Я не уверен, как именно вы хотите, чтобы он вернулся к вам, поэтому я просто сделал читаемую строку.

Это дает "более подробную информацию" по этому вопросу. Может это то, что ты ищешь

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

Это не всегда работает. Добавление Span в DateTime.MinValue может работать, так как это не учитывает високосные годы и т. д. Если вы добавите годы, месяцы и дни к Age с помощью функций AddYears (), AddMonths и AddDays (), он не всегда будет возвращать Datetime. .Настоящая дата.

Athanasios Kataras 23.10.2013 13:44

Сам временной интервал автоматически учитывает високосные годы между двумя датами, поэтому я не уверен, о чем вы. Я спросил на форумах Microsoft, и Microsoft подтвердила, что учитывает високосные годы между двумя датами.

Jacqueline Loriault 23.10.2013 23:29

Рассмотрим следующие ДВА сенария. 1st DateTime.Now - 01.01.2001, а ребенок родился 01.01.2000. 2000 год является високосным, и результатом будет 1 год, 0 месяцев и 1 день. Во втором значении DateTime.Now - 01.01.2002, а ребенок родился 01.01.2001. В этом случае результатом будет 1 год, 0 месяцев и 0 дней. Это произойдет, потому что вы добавляете временной интервал в невисокосный год. Если DateTime.MinValue был високосным годом, то результаты были бы 1 год в первый и 0 лет 11 месяцев и 30 дней. (Попробуйте в своем коде).

Athanasios Kataras 24.10.2013 14:31

Голосовать за! Я придумал решение, которое практически идентично (я использовал DateTime.MinValue.AddTicks (span.Ticks) вместо +, но результат тот же, и у вас код на несколько символов меньше).

Makotosan 17.03.2015 23:57

@AthanasiosKataras, я не понимаю ваш комментарий «Если бы DateTime.MinValue был високосным годом ...». Когда DateTime.MinValue будет високосным годом? Это всегда будет «01.01.10001», что не является високосным годом.

Makotosan 18.03.2015 00:00

Вы совершенно правы, это не так. Но если бы это было так, то был бы результат. Почему это имеет значение? Это не так. В любом случае прыжок или нет, тогда есть примеры, когда это не работает. Это то, что я хотел показать. DIFF правильный. Пролет учитывает високосные годы. Но ДОБАВЛЕНИЯ к базовой дате - нет. Попробуйте примеры в коде, и вы убедитесь, что я прав.

Athanasios Kataras 18.03.2015 21:20

Чтобы рассчитать возраст с ближайшим возрастом:

var ts = DateTime.Now - new DateTime(1988, 3, 19);
var age = Math.Round(ts.Days / 365.0);

не обязательно верно. Думаю, правильнее было бы разделить на 365,25, чтобы как-то учесть високосные годы.

Kat Lim Ruiz 20.03.2014 10:26

Этот классический вопрос заслуживает решения Время Нода.

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

Использование:

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

Вас также могут заинтересовать следующие улучшения:

  • Передача часов как IClock вместо использования SystemClock.Instance улучшит тестируемость.

  • Целевой часовой пояс, скорее всего, изменится, поэтому вам также понадобится параметр DateTimeZone.

См. Также мое сообщение в блоге на эту тему: Дни рождения и другие юбилеи

У меня есть индивидуальный метод расчета возраста, а также сообщение о подтверждении бонуса на случай, если это поможет:

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
}

Вызов метода здесь и передача значения datetime (MM / dd / yyyy, если сервер установлен на локаль США). Замените это любым отображаемым окном сообщения или любым контейнером:

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

Помните, что вы можете форматировать сообщение как хотите.

Однострочный ответ:

DateTime dateOfBirth = Convert.ToDateTime("01/16/1990");
var age = ((DateTime.Now - dateOfBirth).Days) / 365;

А как насчет високосных лет?

nivs1978 15.12.2020 13:50

Это один из наиболее точных ответов, который позволяет определить дату рождения 29 февраля по сравнению с любым годом 28 февраля.

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

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

    return age;
}




Это сегодня! (Следующий - через четыре года.)

Peter Mortensen 29.02.2020 04:06

Вместо этого вы можете использовать DateTime.Today, поскольку время не имеет значения для расчета

tif 18.08.2020 14:04

Просто потому, что я не думаю, что главный ответ так однозначен:

public static int GetAgeByLoop(DateTime birthday)
{
    var age = -1;

    for (var date = birthday; date < DateTime.Today; date = date.AddYears(1))
    {
        age++;
    }

    return age;
}

Проверь это:

TimeSpan ts = DateTime.Now.Subtract(Birthdate);
age = (byte)(ts.TotalDays / 365.25);

Вот функция, которая мне хорошо служит. Никаких расчетов, очень просто.

    public static string ToAge(this DateTime dob, DateTime? toDate = null)
    {
        if (!toDate.HasValue)
            toDate = DateTime.Now;
        var now = toDate.Value;

        if (now.CompareTo(dob) < 0)
            return "Future date";

        int years = now.Year - dob.Year;
        int months = now.Month - dob.Month;
        int days = now.Day - dob.Day;

        if (days < 0)
        {
            months--;
            days = DateTime.DaysInMonth(dob.Year, dob.Month) - dob.Day + now.Day;
        }

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


        return string.Format("{0} year(s), {1} month(s), {2} days(s)",
            years,
            months,
            days);
    }

А вот модульный тест:

    [Test]
    public void ToAgeTests()
    {
        var date = new DateTime(2000, 1, 1);
        Assert.AreEqual("0 year(s), 0 month(s), 1 days(s)", new DateTime(1999, 12, 31).ToAge(date));
        Assert.AreEqual("0 year(s), 0 month(s), 0 days(s)", new DateTime(2000, 1, 1).ToAge(date));
        Assert.AreEqual("1 year(s), 0 month(s), 0 days(s)", new DateTime(1999, 1, 1).ToAge(date));
        Assert.AreEqual("0 year(s), 11 month(s), 0 days(s)", new DateTime(1999, 2, 1).ToAge(date));
        Assert.AreEqual("0 year(s), 10 month(s), 25 days(s)", new DateTime(1999, 2, 4).ToAge(date));
        Assert.AreEqual("0 year(s), 10 month(s), 1 days(s)", new DateTime(1999, 2, 28).ToAge(date));

        date = new DateTime(2000, 2, 15);
        Assert.AreEqual("0 year(s), 0 month(s), 28 days(s)", new DateTime(2000, 1, 18).ToAge(date));
    }

Это может быть так просто:

int age = DateTime.Now.AddTicks(0 - dob.Ticks).Year - 1;

Я бы просто сделал это:

DateTime birthDay = new DateTime(1990, 05, 23);
DateTime age = DateTime.Now - birthDay;

Таким образом вы можете рассчитать точный возраст человека с точностью до миллисекунды, если хотите.

Это не правильно. В вашем коде возраст будет TimeSpan. Не DateTime.

Edward Olamisan 02.10.2015 19:57

Проблема в том, что такой возраст, как «17 лет», не переводится напрямую в TimeSpan, потому что вы не знаете, какие из этих 17 лет были високосными.

Arturo Torres Sánchez 06.10.2015 18:17

Я использовал следующее для этой проблемы. Я знаю, что это не очень элегантно, но работает.

DateTime zeroTime = new DateTime(1, 1, 1);
var date1 = new DateTime(1983, 03, 04);
var date2 = DateTime.Now;
var dif = date2 - date1;
int years = (zeroTime + dif).Year - 1;
Log.DebugFormat("Years -->{0}", years);

public string GetAge(this DateTime birthdate, string ageStrinFormat = null)
{
    var date = DateTime.Now.AddMonths(-birthdate.Month).AddDays(-birthdate.Day);
    return string.Format(ageStrinFormat ?? "{0}/{1}/{2}",
        (date.Year - birthdate.Year), date.Month, date.Day);
}

Это самый простой способ ответить на этот вопрос одной строкой.

DateTime Dob = DateTime.Parse("1985-04-24");
 
int Age = DateTime.MinValue.AddDays(DateTime.Now.Subtract(Dob).TotalHours/24 - 1).Year - 1;

Это также работает для високосных лет.

ваш ответ неверен на один день, он даст день рождения на день раньше

Aman 08.12.2016 17:13

=== Распространенное высказывание (от месяцев до лет) ===

Если вы просто для общего пользования, вот код в качестве информации:

DateTime today = DateTime.Today;
DateTime bday = DateTime.Parse("2016-2-14");
int age = today.Year - bday.Year;
var unit = "";

if (bday > today.AddYears(-age))
{
    age--;
}
if (age == 0)   // Under one year old
{
    age = today.Month - bday.Month;

    age = age <= 0 ? (12 + age) : age;  // The next year before birthday

    age = today.Day - bday.Day >= 0 ? age : --age;  // Before the birthday.day

    unit = "month";
}
else {
    unit = "year";
}

if (age > 1)
{
    unit = unit + "s";
}

Результат теста, как показано ниже:

The birthday: 2016-2-14

2016-2-15 =>  age=0, unit=month;
2016-5-13 =>  age=2, unit=months;
2016-5-14 =>  age=3, unit=months; 
2016-6-13 =>  age=3, unit=months; 
2016-6-15 =>  age=4, unit=months; 
2017-1-13 =>  age=10, unit=months; 
2017-1-14 =>  age=11, unit=months; 
2017-2-13 =>  age=11, unit=months; 
2017-2-14 =>  age=1, unit=year; 
2017-2-15 =>  age=1, unit=year; 
2017-3-13 =>  age=1, unit=year;
2018-1-13 =>  age=1, unit=year; 
2018-1-14 =>  age=1, unit=year; 
2018-2-13 =>  age=1, unit=year; 
2018-2-14 =>  age=2, unit=years; 

Версия 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  

Вау, мне пришлось здесь дать свой ответ ... На такой простой вопрос столько ответов.

private int CalcularIdade(DateTime dtNascimento)
    {
        var nHoje = Convert.ToInt32(DateTime.Today.ToString("yyyyMMdd"));
        var nAniversario = Convert.ToInt32(dtNascimento.ToString("yyyyMMdd"));

        double diff = (nHoje - nAniversario) / 10000;

        var ret = Convert.ToInt32(Math.Truncate(diff));

        return ret;
    }

private int GetYearDiff(DateTime start, DateTime end)
{
    int diff = end.Year - start.Year;
    if (end.DayOfYear < start.DayOfYear) { diff -= 1; }
    return diff;
}
[Fact]
public void GetYearDiff_WhenCalls_ShouldReturnCorrectYearDiff()
{
    //arrange
    var now = DateTime.Now;
    //act
    //assert
    Assert.Equal(24, GetYearDiff(new DateTime(1992, 7, 9), now)); // passed
    Assert.Equal(24, GetYearDiff(new DateTime(1992, now.Month, now.Day), now)); // passed
    Assert.Equal(23, GetYearDiff(new DateTime(1992, 12, 9), now)); // passed
}

Я часто считаю по пальцам. Мне нужно посмотреть в календарь, чтобы понять, когда что-то изменится. Вот что я бы сделал в своем коде:

int AgeNow(DateTime birthday)
{
    return AgeAt(DateTime.Now, birthday);
}

int AgeAt(DateTime now, DateTime birthday)
{
    return AgeAt(now, birthday, CultureInfo.CurrentCulture.Calendar);
}

int AgeAt(DateTime now, DateTime birthday, Calendar calendar)
{
    // My age has increased on the morning of my
    // birthday even though I was born in the evening.
    now = now.Date;
    birthday = birthday.Date;

    var age = 0;
    if (now <= birthday) return age; // I am zero now if I am to be born tomorrow.

    while (calendar.AddYears(birthday, age + 1) <= now)
    {
        age++;
    }
    return age;
}

Выполнение этого в LINQPad дает следующее:

PASSED: someone born on 28 February 1964 is age 4 on 28 February 1968
PASSED: someone born on 29 February 1964 is age 3 on 28 February 1968
PASSED: someone born on 31 December 2016 is age 0 on 01 January 2017

Код в LINQPad - здесь.

Простой код

 var birthYear=1993;
 var age = DateTime.Now.AddYears(-birthYear).Year;

Чтобы посчитать, сколько лет человеку,

DateTime dateOfBirth;

int ageInYears = DateTime.Now.Year - dateOfBirth.Year;

if (dateOfBirth > today.AddYears(-ageInYears )) ageInYears --;

вы только что скопировали первый ответ, что дает Патель?

sensei 10.06.2019 21:45

Просто используйте:

(DateTime.Now - myDate).TotalHours / 8766.0

Текущая дата - myDate = TimeSpan, получите общее количество часов и разделите на общее количество часов в году и получите точный возраст / месяцы / дни ...

А как насчет високосных лет?

Wiktor Zychla 27.06.2018 11:32

Вот самый простой способ рассчитать возраст человека. Подсчитать возраст человека довольно просто, и вот как это сделать! Чтобы код работал, вам нужен объект DateTime с именем BirthDate, содержащий дату рождения.

 C#
        // get the difference in years
        int years = DateTime.Now.Year - BirthDate.Year; 
        // subtract another year if we're before the
        // birth day in the current year
        if (DateTime.Now.Month < BirthDate.Month || 
            (DateTime.Now.Month == BirthDate.Month && 
            DateTime.Now.Day < BirthDate.Day)) 
            years--;
  VB.NET
        ' get the difference in years
        Dim years As Integer = DateTime.Now.Year - BirthDate.Year
        ' subtract another year if we're before the
        ' birth day in the current year
        If DateTime.Now.Month < BirthDate.Month Or (DateTime.Now.Month = BirthDate.Month And DateTime.Now.Day < BirthDate.Day) Then 
            years = years - 1
        End If

«Вот самый простой способ рассчитать чей-то возраст». Это действительно не самый простой способ. Используя Noda Time, это просто int years = Period.Between(birthDate, today).Years;.

Jon Skeet 08.02.2018 18:13

@JonSkeet Честно говоря, Noda Time не является частью какого-либо стандарта .Net. Учитывая это, этот является один из самых простых способов рассчитать возраст человека в годах.

Clearer 23.04.2018 12:42

var birthDate = ... // DOB
var resultDate = DateTime.Now - birthDate;

Используя resultDate, вы можете применять свойства TimeSpan, как хотите.

Очень простой ответ

        DateTime dob = new DateTime(1991, 3, 4); 
        DateTime now = DateTime.Now; 
        int dobDay = dob.Day, dobMonth = dob.Month; 
        int add = -1; 
        if (dobMonth < now.Month)
        {
            add = 0;
        }
        else if (dobMonth == now.Month)
        {
            if (dobDay <= now.Day)
            {
                add = 0;
            }
            else
            {
                add = -1;
            }
        }
        else
        {
            add = -1;
        } 
        int age = now.Year - dob.Year + add;

Должно быть описание вашего ответа для OP, а также для будущих читателей.

Suraj Kumar 05.03.2020 12:04

Ваш код не простой, это беспорядок. Я бы написал это, может быть, вы не верите, но код ниже делает то же самое (сейчас - dob). Год

Võ Quang Hòa 15.04.2020 06:42

int Age = new DateTime((DateTime.Now - BirthDate).Ticks).Year -1;
Console.WriteLine("Age {0}", Age);

Я думаю, что эту проблему можно решить более простым способом -

Класс может быть таким:

using System;

namespace TSA
{
    class BirthDay
    {
        double ageDay;
        public BirthDay(int day, int month, int year)
        {
            DateTime birthDate = new DateTime(year, month, day);
            ageDay = (birthDate - DateTime.Now).TotalDays; //DateTime.UtcNow
        }

        internal int GetAgeYear()
        {
            return (int)Math.Truncate(ageDay / 365);
        }

        internal int GetAgeMonth()
        {
            return (int)Math.Truncate((ageDay % 365) / 30);
        }
    }
}

И звонки могут быть такими:

BirthDay b = new BirthDay(1,12,1990);
int year = b.GetAgeYear();
int month = b.GetAgeMonth();

Я ничего не знаю о DateTime, но могу сделать следующее:

using System;
                    
public class Program
{
    public static int getAge(int month, int day, int year) {
        DateTime today = DateTime.Today;
        int currentDay = today.Day;
        int currentYear = today.Year;
        int currentMonth = today.Month;
        int age = 0;
        if (currentMonth < month) {
            age -= 1;
        } else if (currentMonth == month) {
            if (currentDay < day) {
                age -= 1;
            }
        }
        currentYear -= year;
        age += currentYear;
        return age;
    }
    public static void Main()
    {
        int ageInYears = getAge(8, 10, 2007);
        Console.WriteLine(ageInYears);
    }
}

Немного сбивает с толку, но если присмотреться к коду внимательнее, все будет иметь смысл.

Другие вопросы по теме