Як в C # як я обчислюю чийсь вік на основі дня народження типу DateTime?


1866

Дано a DateTime день народження людини, як я обчислюю їхній вік у роках?


147
що всі відповіді поки що пропустили - це залежить від того, де людина народилася і де вони зараз.
Яур

40
@Yaur: Просто перетворіть час тепер + народження в GMT / UTC, вік - лише відносне значення, отже, часові пояси не мають значення. Для визначення поточного часового поясу користувача можна використовувати GeoLocating.
Стефан Штайгер

Чому б не розглянути [Джуліан Дата] [1]? [1]: stackoverflow.com/questions/7103064 / ...
Мухаммад Hewedy

5
Якщо ми беремо до уваги пропозицію @Yaur щодо обчислень між часовим поясом, чи повинен день переходу на світло в будь-який спосіб впливати на обчислення?
DDM

6
Незважаючи на те, що це, очевидно, домашнє завдання, і жодна наявна спроба не була надана.
Марі

Відповіді:


2122

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

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

Однак це передбачає, що ви шукаєте західну ідею віку та не використовуєте східноазіатський розрахунок .


252
Просто хотів прокоментувати продуктивність DateTime.Now. Якщо вам не потрібно точне значення часового поясу, використовуйте DateTime.UtcNow це набагато швидше.
JAG

104
Зважаючи на те, що ми говоримо про дні народження, ви можете просто використовувати DateTime.Today, коли частина часу не має значення.
Трістан Уорнер-Сміт

78
Ця відповідь працює не з усіма локалями та різними віками. Кілька країн пропустили дати після народження нинішніх живих людей, зокрема Росія (1918), Греція (1924) та Туреччина (1926).
Lars D

30
Власне, це все ще не зовсім правильно. Цей код передбачає, що "bday" є частиною дати DateTime. Це крайній випадок (я думаю, більшість людей просто пропускають дати, а не дати), але якщо ви переходите в день народження як дату, так і час, де час перевищує 00:00:00, тоді ви ' Я наткнуся на помилку, що вказував Данвіл. Налаштування bday = bday.Date виправляє це.
Øyvind

119
Останній рядок змусив мене занадто багато думати. Натомість як щодо: якщо (bday.AddYears (вік)> зараз) вік--; Це здається більш інтуїтивним виразом.
cdiggins

1015

Це дивний спосіб зробити це, але якщо ви відформатуєте дату 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;
}

5
Насправді це чудово для використання в MS-SQL з полями datetime (загальна кількість днів з 01-011900)
Патрік

у своїй альтернативній відповіді ви можете уникнути переливу цілих чисел, віднімаючи роки, потім відніміть місяць * 30,5 + день і розділіть на 366
numerek

5
@numerek Будь ласка, опублікуйте запропоновані зміни як власну відповідь. Для того, що це варто, кількість разів 10000 поточного року ніде не є цілим переливом, на два порядки. 20,150,000 проти 2,147,483,648
GalacticCowboy

7
@LongChalk 20180101 - 20171231 = 8870. Закиньте останні 4 цифри, і ви матимете на увазі 0вік. Як ви потрапили 1?
Руфус Л

4
Я знаю, що це стара відповідь, але я не став би використовувати метод розширення з неї, це не правильне місце для визначення таких логік.
Лукка Феррі

391

Ось фрагмент тесту:

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

33
Хоча цей код працює, він стверджує, що людина, яка народилася у високосний день, досягає наступного віку 1 березня у нестримні роки, а не 28 лютого. Насправді будь-який варіант може бути правильним . У Вікіпедії є про що сказати . Тож ваш код не є "неправильним", і рішення не приймається.
Метт Джонсон-Пінт

18
@MattJohnson Я думаю, що це насправді правильно. Якщо моїм днем ​​був 29 лютого, то 28 лютого мій день не минув, і я все одно повинен бути такого ж віку, як і 27 лютого. Однак 1 березня ми пройшли мій день, і я повинен був бути наступним віком. У США в бізнесі, який продає алкоголь, з'явиться знак, який говорить про те, як "Якщо ти народився після цього дня в РРРР, ти не можеш придбати алкоголь" (де РГРР змінюється щороку). Це означає, що хтось, народжений 29 лютого, не може придбати алкоголь 28 лютого року, коли їм виповниться 21 рік (більшість місць), і надає підтримку думці, що до 1 березня вони не на рік старші
jfren484

4
@ jfren484 - читайте статтю у Вікіпедії. Він значно відрізняється в різних юрисдикціях.
Метт Джонсон-Пінт

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

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

109

Проста відповідь на це - застосувати AddYears як показано нижче, оскільки це єдиний нативний метод, який додає роки до 29 лютого високосних років та отримає правильний результат 28 лютого за загальні роки.

Деякі вважають, що 1 березня є днем ​​народження молодняку, але ні. Ні, ні якесь офіційне правило це не підтверджує, а також загальна логіка не пояснює, чому деякі народжені в лютому мають мати 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();
    }
}

Найважливіший приклад дати:

Дата народження: 2000-02-29 Пізніша дата: 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
}

4
Коментар щодо того, що день народження 29-го лютого відбувся 1-го березня, технічно мати його 28-го зарано (фактично 1 день раніше). На 1-й день занадто пізно. Але оскільки день народження є між, використання першого для обчислення віку у нестримні роки має для мене більше сенсу, оскільки ця людина справді є такою старшою 1-го березня (та 2-го та 3-го) кожного року, але не 28 лютого.
CyberClaw

1
З точки зору розробки програмного забезпечення, писати це як метод розширення не має для мене особливого сенсу. date.Age(other)?
марш

90

Моя пропозиція

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

Здається, рік змінюється в потрібну дату. (Я помітив тест до віку 107.)


26
Я не думаю, що Гаррі Патч оцінив би вашу методологію точкового тестування: latimes.com/news/obituaries/…
MusiGenesis

3
Google кажеdays in a year = 365.242199
10.10

12
Середня тривалість року за григоріанським календарем становить 365,2425 днів.
dan04

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

13
^^ Тому що іноді це важливо. У моєму тестуванні це не вдається в день народження людей, він повідомляє про них молодше, ніж вони є.
ChadT

76

Ще одна функція, не мною, але знайдена в Інтернеті та трохи вдосконалена:

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

53

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

Рішення для двох - це не таке точне визначення загального віку, але сприймається людьми як точне. Люди також зазвичай користуються ним, коли вони обчислюють свій вік "вручну":

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, оскільки вони зміщують кількість днів у високосні роки
  • Я розмістив там трохи більше рядків для читабельності

Ще одна примітка ... Я створив би два статичні перевантажені методи, один для універсального використання, другий для зручності використання:

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

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

50

Ось один вкладиш:

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

23
Це порушено. Здійснено перевірку: публічний статичний int CalculateAge (DateTime dateOfBirth, DateTime dateToCalculateAge) {return new DateTime (dateToCalculateAge.Subtract (dateOfBirth) .Ticks) .Year - 1; } ... Дає вік 14 років, коли я ввожу 1990-06-01 і обчислюю вік у день ДО 14-го дня народження (1990-05-31).
Kjensen

43

Це версія, яку ми тут використовуємо. Це працює, і це досить просто. Це та сама ідея, що і у Джеффа, але, я думаю, це трохи зрозуміліше, оскільки вона відокремлює логіку віднімання, тому зрозуміти її трохи простіше.

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

Ви можете розширити потрійного оператора, щоб зробити його ще зрозумілішим, якщо ви думаєте, що таке щось незрозуміле.

Очевидно, це робиться як метод розширення DateTime, але чітко ви можете захопити той рядок коду, який виконує роботу, і помістити його в будь-яке місце. Тут ми маємо ще одне перевантаження методу Extension, який проходить DateTime.Now, просто для повноти.


6
Я думаю, що це може бути вимкнено одним днем, коли точно один з дати DateFfirthth або DateAsAt припадає на високосний рік. Розглянемо вік людини, народженої 1 березня 2003 року 29 лютого 2004 року. Щоб виправити це, потрібно провести лексикографічне порівняння пар (Month, DayOfMonth) і використовувати його для умовного.
Дуг МакКлін

1
це також не буде показувати правильний вік, як ваш день народження.
dotjoe

43

Найкращий спосіб, який я знаю через високосні роки, і все:

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

34

Я використовую це:

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

32

Це дає «детальніше» цьому питанню. Можливо, це те, що ви шукаєте

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

1
Це не працює весь час. Додавання прольоту до DateTime.MinValue могло б працювати, бо це не враховує високосні роки тощо. Якщо ви додасте роки, місяці та дні до віку за допомогою функції AddYears (), AddMonths та AddDays (), вона не завжди повертає Datetime . Тепер дата.
Атанасіос Катарас

3
Сам часовий проміжок автоматично враховує високосні роки між двома датами, тож я не впевнений, про що йдеться. Я запитав на форумах мікрософт, і microsoft підтвердив, що він враховує високосні роки між двома датами.
Жаклін Лоріа

2
Розглянемо наступні ДВІ сенарії. 1-й Дата. Час. Зараз 1.01.2001 р., 1.01.2000 р. Народжується дитина. 2000 рік - високосний рік, результатом буде 1 рік, 0 місяців та 1 день. У другому сенаріоні DateTime.Now - 1/1/2002, і дитина народжується 01.01.2001. У цьому випадку результат становитиме 1 рік, 0 місяців та 0 днів. Це станеться тому, що ви додаєте часовий проміжок у не високосний рік. Якщо DateTime.MinValue був високосним, то результати були б 1 рік на першій, а 0 років - 11 місяців та 30 днів. (Спробуйте це у своєму коді).
Атанасіос Катарас

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

4
Ви абсолютно праві, це не так. Але якби це був результат. Чому це важливо? Це не так. У будь-якому випадку стрибок чи ні, є приклади, коли це не працює. Це я хотів показати. DIFF правильний. Проміжок враховує високосні роки. Але додавання до базової дати це не так. Спробуйте приклади в коді, і ви побачите, що я маю рацію.
Афанасіос Катарас

28

Я створив функцію, визначену користувачем 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;
    }
};

28

Ось ще одна відповідь:

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 день.

Пояснення, чому це працює ( знято звідси ), є:

Давайте встановимо 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

Ми знаємо, що нам потрібно або Yn-Ybякщо дата вже досягнута,Yn-Yb-1 якщо її немає.

а) Якщо Mn<Mb, маємо-341 <= 31*(Mn-Mb) <= -31 and -30 <= Dn-Db <= 30

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

З цілим поділом

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

б) Якщо Mn=MbіDn<Db ми маємо31*(Mn - Mb) = 0 and -30 <= Dn-Db <= -1

З цілим поділом, знову ж таки

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

в) Якщо Mn>Mb, маємо31 <= 31*(Mn-Mb) <= 341 and -30 <= Dn-Db <= 30

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

З цілим поділом

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

г) Якщо Mn=Mbі Dn>Db, маємо 31*(Mn - Mb) = 0 and 1 <= Dn-Db <= 30

З цілим поділом, знову ж таки

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

д) Якщо Mn=MbіDn=Db ми маємо31*(Mn - Mb) + Dn-Db = 0

і таким чином (31*(Mn - Mb) + (Dn - Db)) / 372 = 0


3
Я натрапив на цю довгу і надокучливу дискусію, і ваше рішення - це дійсно приємний і невеликий підхід. Дякуємо за те, що це просто
набуходоносор

25

Я витратив деякий час, працюючи над цим, і придумав це, щоб обчислити чийсь вік у роках, місяцях та днях. Я протестував проти проблеми 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++;
    }

}

21

Чи потрібно розглядати людей, які менше 1 року? як китайська культура, ми описуємо вік маленьких дітей як 2 місяці або 4 тижні.

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

Ця реалізація пройшла нижче тестових випадків.

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

Сподіваюся, що це корисно.


20

Зберігати це просто (і, можливо, дурно :)).

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

19

Найпростіший спосіб, який я коли-небудь знайшов, це такий. Він працює коректно для місцевості США та Західної Європи. Не можу спілкуватися з іншими місцевостями, особливо з такими місцями, як Китай. 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;
}

Слід зазначити, що цей код передбачає:

  • Західне (європейське) врахування віку та
  • Календар, як григоріанський календар, який вставляє єдиний високосний день наприкінці місяця.

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

Я не впевнений, як саме ви хотіли б, щоб він повернувся до вас, тому я просто зробив читабельну рядок.


18

Це не пряма відповідь, а скоріше філософські міркування про проблему, що існує, з квазінаукової точки зору.

Я заперечую, що питання не визначає одиницю та культуру, за якою вимірювати вік, більшість відповідей, мабуть, передбачає ціле щорічне представлення. Одиниця СІ за часом - це 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;

Цікаво, як довго нам потрібно пройти до того, як релятивістський вік за лічені секунди стане кориснішим, ніж грубе наближення циклів Земля навколо Сонця протягом життя досі :) Або іншими словами, коли періоду потрібно вказати місцеположення або функція, що представляє рух для себе дійсним :)


17

Ось рішення.

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


16

Це одна з найточніших відповідей, яка здатна вирішити день народження 29 лютого порівняно з будь-яким роком 28 лютого.

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

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

    return age;
}




Це сьогодні! (Наступний - через чотири роки.)
Пітер Мортенсен

15

У мене є індивідуальний метод обчислення віку, плюс повідомлення про підтвердження бонусу на випадок, якщо це допомагає:

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
}

Зателефонуйте сюди і видайте значення дати (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

Пам'ятайте, що ви можете форматувати повідомлення будь-яким способом.


14

Як щодо цього рішення?

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

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

10

Наступний підхід (витяг з бібліотеки часових періодів для .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

10

Це класичне питання заслуговує на час Noda .

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 параметр.

Дивіться також мій пост у щоденнику на цю тему: Поводження з дня народження та інших річниць


Ви пов’язані з Noda Time?
Зімано

Я зробив внесок у це, але це, перш за все, Джон Скіт.
Метт Джонсон-

9

Я використовував рішення 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;
}

9

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

8

Я зробив одну невелику зміну до Марка Сона Сона: я переписала третій рядок, щоб вираз можна було проаналізувати трохи легше.

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

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

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.