Как рассчитать чей-то возраст в C #?

1656

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

  • 127
    то, что все ответы до сих пор упущены, это то, что это зависит от того, где человек родился и где он сейчас находится.
  • 37
    @Yaur: Просто переведите время текущего + рождения в GMT / UTC, возраст является только относительным значением, поэтому часовые пояса не имеют значения. Для определения текущего часового пояса пользователя вы можете использовать GeoLocating.
Показать ещё 5 комментариев
Теги:
datetime

63 ответа

1759
Лучший ответ

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

// Save today date.
var today = DateTime.Today;
// Calculate the age.
var age = today.Year - birthdate.Year;
// Go back to the year the person was born in case of a leap year
if (birthdate > today.AddYears(-age)) age--;

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

  • 224
    Просто хотел прокомментировать производительность DateTime.Now. Если вам не нужно точное значение часового пояса, используйте DateTime.UtcNow, это намного быстрее.
  • 92
    Поскольку мы говорим о днях рождения, вы можете просто использовать DateTime.Today, учитывая, что часть времени не имеет значения.
Показать ещё 33 комментария
954

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

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

20080814 - 19800703 = 280111 

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

Код С#:

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;
}
  • 50
    Извините за yyyymmdd . Первый продукт базы данных я использовал на C omputer P ersonal (около 1981) хранятся даты в этом формате. Любые манипуляции с датами были намного проще.
  • 2
    На самом деле это отлично подходит для использования на MS-SQL с полями datetime (всего дней с 01-011900)
Показать ещё 12 комментариев
332

Я не знаю, как можно принять неправильное решение. Правильный фрагмент С# был написан Майклом Стамом

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

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

Здесь у вас есть методы:

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;
}
  • 8
    Выход был -Test 9 9 8
  • 28
    Хотя этот кодекс работает, он утверждает, что человек, родившийся в високосный день, достигает следующего года 1 марта в не високосные годы, а не 28 февраля. На самом деле, любой из этих вариантов может быть правильным . Википедия есть что сказать по этому поводу . Таким образом, хотя ваш код не является «неправильным», также не является принятым решением.
Показать ещё 12 комментариев
116

Я не думаю, что какой-либо из ответов до сих пор предоставляет культурам, которые вычисляют возраст по-разному. См., Например, Восточно-азиатское расценение возраста против того, что на Западе.

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

  • 24
    Из статьи Википедии, которую вы предоставили: «В Китае и Японии она используется для традиционных гаданий или религии, и она исчезает в повседневной жизни между людьми в городе».
  • 7
    @ Некоторые - Корейцы до сих пор используют эту систему в первую очередь.
Показать ещё 7 комментариев
95

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

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

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

  • Элемент списка

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 Поздняя дата: 2011-02-28 Возраст: 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
}
  • 3
    Комментарий о том, чтобы иметь день рождения 29 февраля 1 марта, технически, иметь его 28 марта слишком рано (на самом деле на 1 день раньше). На 1-й день уже поздно. Но поскольку день рождения промежуточный, использование 1-го числа для расчета возраста в не високосные годы имеет для меня больше смысла, поскольку этот человек действительно 1 лет (и 2-го и 3-го) действительно каждый год, но не 28-го февраля.
  • 0
    С точки зрения разработки программного обеспечения, писать это как метод расширения не имеет особого смысла для меня. date.Age(other) ?
74

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

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

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

  • 4
    Откуда берется 365.255? Я не думаю, что это будет работать в целом.
  • 13
    365 дней в году. +0.25 за високосные годы. +0,005 за другие исправления
Показать ещё 8 комментариев
62

Другая функция, а не я, но найденная в Интернете, немного уточнила:

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

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

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

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

Я опаздываю на вечеринку, но здесь однострочный:

int age = new DateTime(DateTime.Now.Subtract(birthday).Ticks).Year-1;
  • 22
    Сломано. Сделано для тестирования: public static int CalculateAge (DateTime dateOfBirth, DateTime dateToCalculateAge) {вернуть новый DateTime (dateToCalculateAge.Subtract (dateOfBirth) .Ticks) .Year - 1; } ... Дает 14 лет, когда я ввожу 1990-06-01 и вычисляю возраст в день ДО 14-летия (1990-05-31).
32

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

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

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

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

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

Много лет назад, чтобы предоставить трюк калькулятора возраста на моем веб-сайте, я написал функцию для вычисления возраста до доли. Это быстрый порт этой функции для С# (из версия PHP). Боюсь, я не смог проверить версию С#, но надеюсь, что вам все равно понравится!

(По общему признанию, это немного неудобно для показа пользовательских профилей в Stack Overflow, но, возможно, читатели найдут для этого некоторое использование.: -))

double AgeDiff(DateTime date1, DateTime date2) {
    double years = date2.Year - date1.Year;

    /*
     * If date2 and date1 + round(date2 - date1) are on different sides
     * of 29 February, then our partial year is considered to have 366
     * days total, otherwise it 365. Note that 59 is the day number
     * of 29 Feb.
     */
    double fraction = 365
            + (DateTime.IsLeapYear(date2.Year) && date2.DayOfYear >= 59
            && (date1.DayOfYear < 59 || date1.DayOfYear > date2.DayOfYear)
            ? 1 : 0);

    /*
     * The only really nontrivial case is if date1 is in a leap year,
     * and date2 is not. So let handle the others first.
     */
    if (DateTime.IsLeapYear(date2.Year) == DateTime.IsLeapYear(date1.Year))
        return years + (date2.DayOfYear - date1.DayOfYear) / fraction;

    /*
     * If date2 is in a leap year, but date1 is not and is March or
     * beyond, shift up by a day.
     */
    if (DateTime.IsLeapYear(date2.Year)) {
        return years + (date2.DayOfYear - date1.DayOfYear
                - (date1.DayOfYear >= 59 ? 1 : 0)) / fraction;
    }

    /*
     * If date1 is not on 29 February, shift down date1 by a day if
     * March or later. Proceed normally.
     */
    if (date1.DayOfYear != 59) {
        return years + (date2.DayOfYear - date1.DayOfYear
                + (date1.DayOfYear > 59 ? 1 : 0)) / fraction;
    }

    /*
     * Okay, here date1 is on 29 February, and date2 is not on a leap
     * year. What to do now? On 28 Feb in date2 year, the ``age''
     * should be just shy of a whole number, and on 1 Mar should be
     * just over. Perhaps the easiest way is to a point halfway
     * between those two: 58.5.
     */
    return years + (date2.DayOfYear - 58.5) / fraction;
}
31

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

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

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

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

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

26

Это дает "более подробно" на этот вопрос. Возможно, это то, что вы ищете

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);
  • 0
    Это не работает все время. Добавление Span к DateTime.MinValue может работать, поскольку это не учитывает високосные годы и т. Д. Если вы добавляете Year, месяцы и дни к Age с помощью функций AddYears (), AddMonths и AddDays (), он не всегда возвращает Datetime .Новая дата.
  • 1
    Сам таймспэн автоматически учитывает високосные годы между двумя датами, поэтому я не знаю, о чем вы. Я спрашивал на форумах Microsoft и Microsoft подтвердил, что он учитывает високосные годы между двумя датами.
Показать ещё 4 комментария
23

Я создал пользовательскую функцию 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;
    }
};
22

Я потратил некоторое время на то, чтобы поработать над этим, и придумал это, чтобы рассчитать кого-то возраст годами, месяцами и днями. Я тестировал проблему 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++;
    }

}
18

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

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 если она не Yn-Yb-1.

a) Если 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

d) Если Mn=Mb и Dn>Db, то 31*(Mn - Mb) = 0 and 1 <= Dn-Db <= 3 0

С целым делением снова

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

e) Если Mn=Mb и Dn=Db, то 31*(Mn - Mb) + Dn-Db = 0

и поэтому (31*(Mn - Mb) + (Dn - Db))/372 = 0

  • 1
    Я наткнулся на эту долгую и раздражающую дискуссию, и ваше решение - действительно хороший и маленький подход. Спасибо за простоту
17
TimeSpan diff = DateTime.Now - birthdayDateTime;
string age = String.Format("{0:%y} years, {0:%M} months, {0:%d}, days old", diff);

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

17

Нужно ли рассматривать людей, которые меньше 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);
}

Надеюсь, что это будет полезно.

17

Сохранение его простым (и, возможно, глупым:)).

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

Самый простой способ, который я когда-либо нашел, - это. Он работает правильно для локалей США и западной Европы. Не может разговаривать с другими местами, особенно в таких местах, как Китай. 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;
}

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

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

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

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

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

    return age;
}
15

Вот решение.

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);
  • 0
    Со строкой concat это было бы возможно: 47 лет 11 мес. 7 дней
14

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

Я бы сказал, что в вопросе не указывается единица или культура, в которой можно измерить возраст, большинство ответов, кажется, принимают целочисленное годовое представление. SI-единица для времени 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;

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

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

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

13

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

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

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

13

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

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

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

Я использовал решение 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 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;
}
7

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

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

Я также сделал это для ясности.

6

Ничего себе, я должен был дать свой комментарий здесь.. Есть так много ответов для такого простого

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

Надеюсь, что это может помочь кому-то, по крайней мере, заставит кого-то подумать..:)

  • 0
    Это должно быть написано на английском языке.
6

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

DateTime Dob = DateTime.Parse("1985-04-24");

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

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

  • 1
    ваш ответ неверен на один день, он даст день рождения за день до
6

Это просто и, кажется, точно для моих нужд. Я делаю предположение для целей високосных лет, что независимо от того, когда человек решит отпраздновать день рождения, они не технически на год старше, пока не прошло 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;

Сообщите нам, если вы обнаружите какие-либо проблемы;)

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

=== Общее высказывание (от нескольких месяцев до нескольких лет) ===

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

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

Почему это не может быть так просто?

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

Этот классический вопрос заслуживает решения Noda Time.

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

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

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

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

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

    return (int) period.Years;
}

Применение:

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

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

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

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

Смотрите также мое сообщение в блоге на эту тему: Работа с деньгами и другие юбилеи

  • 0
    Вы связаны с Noda Time?
  • 0
    Я сделал вклад в это, но это прежде всего Джон Скит.
4

Здесь используется расширитель 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;
    }
}
  • 3
    тьфу, не делай этого. ToString и int.Parse относительно дороги, и хотя я против микро-оптимизации, скрывая дорогие функции в методах расширения, которые должны быть тривиальными операциями, не очень хорошая идея.
  • 1
    Кроме того, это дубликат ответа ScArcher2: stackoverflow.com/questions/9/…
Показать ещё 1 комментарий
3
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);
}
2
var birthDate = ... // DOB
var resultDate = DateTime.Now - birthDate;

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

2

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

(DateTime.Now - myDate).TotalHours / 8766.0

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

  • 0
    Как насчет високосных лет?
2

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

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 здесь

2

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

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

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

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

И вот unit test:

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

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

Вот небольшой пример кода для С# I, с которым вы столкнулись, будьте осторожны с краями, особенно в високосные годы, но не все вышеупомянутые решения учитывают их. Нажатие ответа как 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;
}
  • 1
    Мне больше всего нравится это решение, однако при подсчете месяцев оно должно быть if (months> = 12). Попробуйте 6-8-2012 - 6-4-1993, чтобы проверить.
2

Будет ли это работать?

public override bool IsValid(DateTime value)
{
    _dateOfBirth =  value;
    var yearsOld = (double) (DateTime.Now.Subtract(_dateOfBirth).TotalDays/365);
    if (yearsOld > 18)
        return true;
    return false; 
}
  • 0
    Отрицательный, пожалуйста, объясните причину !!!
  • 5
    Вот это да. Почему значение является объектом, а не DateTime? Подпись метода должна быть public override bool Is18OrOlder(DateTime birthday) Как насчет людей, которые родились 29 февраля? Кто сказал, что мы пытаемся проверить, было ли пользователю хотя бы 18 лет? Вопрос был "как я могу рассчитать чей-то возраст?"
Показать ещё 2 комментария
2

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

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

textBox1.Text = difference.Years + " years " + difference.Months + " Months " + difference.Days + " days";
  • 5
    Нету. TimeSpan как дни, но не месяцы или годы
1

Вот простейший способ рассчитать возраст человека.
Вычисление кого-то возраста довольно просто, и вот как! Для того, чтобы код работал, вам нужен объект 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
  • 4
    «Вот самый простой способ подсчитать чей-то возраст». Это действительно не самый простой способ. Используя Noda Time, это просто int years = Period.Between(birthDate, today).Years; ,
  • 1
    @JonSkeet Честно говоря, Noda Time не является частью какого-либо стандарта .Net. Учитывая , что это один из самых простых способа вычислить возраст человека в годах.
1

Простой код

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

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

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

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

  • 1
    Это не правильно. В вашем коде возраст будет TimeSpan. Не DateTime.
  • 0
    Проблема в том, что такой возраст, как «17 лет», не переводится непосредственно в интервал времени, потому что вы не знаете, какие из этих 17 лет были високосными.
1

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

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

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

    return age;
}
1

Почему помощь MSDN не говорила вам об этом? Это выглядит так очевидно:

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?
  • 6
    Что ж, это здорово, если вы находитесь в одной из нулевых стран на Земле, где возраст взрослых людей измеряется днями.
1

С меньшим количеством конверсий и 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
        );
}
1

Вот очень простой и удобный пример.

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

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

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

Я создал структуру 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 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.
}
0

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

DateTime dateOfBirth;

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

if (dateOfBirth > today.AddYears(-ageInYears )) ageInYears --;
  • 1
    Это будет означать, что кому-то, родившемуся 31 декабря 2000 года, исполнилось 18 лет 1 января 2018 года! 2018-2000 = 18
  • 0
    Спасибо, что напомнили мне, что я обновил свой ответ сейчас, в то время я забыл добавить последнюю строку. Так что спасибо, что поправили меня. :)
-1

Проверьте это:

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

Один линейный ответ,

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

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

var ts = DateTime.Now - new DateTime(1988, 3, 19);
var age = Math.Round(ts.Days / 365.0);
  • 1
    не обязательно правда. Я думаю, что правильным было бы разделить на 365,25 для учета високосных лет как-то
-4
int age = DateTime.Now.Year - birthday.Year;
if (DateTime.Now.Month < birthday.Month || DateTime.Now.Month == birthday.Month 
&& DateTime.Now.Day < birthday.Day) age--;
  • 0
    Нет, это не так просто. Не в списке. Вы также, кажется, не очень хорошо поняли вопрос.
  • 0
    На самом деле, это не правильно. Ваше условие неуклюже и на самом деле не работает правильно. Должно быть так: if (DateTime.Now.Month <= birthday.Month && DateTime.Now.Day < birthday.Day) я удалил ваше среднее состояние; в сочетании < и == . То, как вы это сделали, не проверяет дату; || значение true приведет к тому, что && даже не будет оценено.
Показать ещё 1 комментарий
-6

Мне не нравятся многие ответы здесь, потому что они берут несколько строк кода, чтобы сделать то, что должно быть очень простым вычислением datemath (пожалуйста, сохраните все свои комментарии о расчете возраста в других культурах, если вы не хотите опубликовать ответ, который охватывает их). Мой один лайнер, используя простые функции datemath и math, которые существуют в С#, sqlserver, mysql и т.д.:

year(@today)-year(@birthDate)+floor((month(@today)-month(@birthdate)+floor((day(@today)-day(@birthdate))/31))/12)

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

Ещё вопросы

Сообщество Overcoder
Наверх
Меню