Учитывая DateTime, представляющий день рождения человека, как мне рассчитать его возраст в годах?
@Yaur: Просто преобразуйте текущее время + время рождения в GMT / UTC, возраст - это только относительное значение, поэтому часовые пояса не имеют значения. Для определения текущего часового пояса пользователя вы можете использовать GeoLocating.
Почему бы не рассмотреть [Юлианскую дату] [1]? [1]: stackoverflow.com/questions/7103064/…
Если мы принимаем во внимание предложение @Yaur о расчетах между часовыми поясами, должно ли летнее время влиять на расчет каким-либо образом?
Обратите внимание, что для лиц младше одного года их возраст указывается в днях, неделях или месяцах. Время перехода для блоков может зависеть от домена.
Как мы все видим, окончательного определения возраста не существует. Многие женщины, которых я встречал, имеют тенденцию округлять свое время жизни до целого года до двадцати с небольшим лет, а затем начинают округлять в меньшую сторону. Я родился 3 января, поэтому я просто вычитаю текущий год из года моего рождения, независимо от того, какой он день. некоторые люди думают, что если вы родились в високосный день, вы стареете в соотношении 1/4. Что, если бы вы родились в високосную секунду? считается ли 8-месячный ребенок за 1? Если я лечу на запад, стану ли я моложе? Если мое сердце остановится на минуту, следует ли мне включить это в расчет?
Сценарий, который не учитывается всеми ответами: мой прадедушка родился 29 февраля 1928 года. Он умер в позапрошлом году, всего за 13 месяцев до своего 23-го дня рождения.
@Rab - вообще говоря, возраст человека, родившегося в високосный год, не рассчитывается таким образом .... но грамматика выше потребовала времени, чтобы выяснить.
Это второй из существующих общедоступных вопросов!





Лучший способ, который я знаю из-за високосных лет и всего остального:
DateTime birthDate = new DateTime(2000,3,1);
int age = (int)Math.Floor((DateTime.Now - birthDate).TotalDays / 365.25D);
Еще одна функция, не мной, но найденная в Интернете и немного доработанная:
public static int GetAge(DateTime birthDate)
{
DateTime n = DateTime.Now; // To avoid a race condition around midnight
int age = n.Year - birthDate.Year;
if (n.Month < birthDate.Month || (n.Month == birthDate.Month && n.Day < birthDate.Day))
age--;
return age;
}
Мне приходят в голову только две вещи: а как насчет людей из стран, которые не используют григорианский календарь? Я думаю, DateTime.Now находится в серверной культуре. У меня абсолютно нулевой опыт работы с азиатскими календарями, и я не знаю, есть ли простой способ конвертировать даты между календарями, но на всякий случай вам интересно узнать об этих китайских парнях из 4660 года :-)
Похоже, что это лучше всего подходит для разных регионов (форматов даты).
Легкое для понимания и простое решение.
// Save today's date.
var today = DateTime.Today;
// Calculate the age.
var age = today.Year - birthdate.Year;
// Go back to the year in which the person was born in case of a leap year
if (birthdate.Date > today.AddYears(-age)) age--;
Однако это предполагает, что вы ищете информацию о возрасте западный, а не используете Восточноазиатский расчет.
Этот ответ не работает для всех регионов и всех возрастов. Некоторые страны пропустили даты после рождения ныне живущих людей, в том числе Россия (1918 г.), Греция (1924 г.) и Турция (1926 г.).
Собственно, это еще не совсем так. Этот код предполагает, что bday - это часть даты DateTime. Это крайний случай (я предполагаю, что большинство людей просто будут передавать даты, а не дату-время), но если вы передадите день рождения как дату и время, где время больше 00:00:00, тогда вы ' Я столкнусь с ошибкой, на которую указал Данвил. Установка bday = bday.Date исправляет это.
Это та версия, которую мы здесь используем. Это работает, и это довольно просто. Это та же идея, что и у Джеффа, но я думаю, что она немного яснее, потому что разделяет логику вычитания единицы, поэтому ее немного легче понять.
public static int GetAge(this DateTime dateOfBirth, DateTime dateAsAt)
{
return dateAsAt.Year - dateOfBirth.Year - (dateOfBirth.DayOfYear < dateAsAt.DayOfYear ? 0 : 1);
}
Вы можете расширить тернарный оператор, чтобы сделать его еще более ясным, если считаете, что что-то в этом роде непонятно.
Очевидно, что это делается как метод расширения на DateTime, но очевидно, что вы можете взять одну строчку кода, которая выполняет эту работу, и поместить ее куда угодно. Здесь у нас есть еще одна перегрузка метода Extension, который передается в DateTime.Now, просто для полноты.
Я думаю, что это может быть отключено на один день, когда в високосный год выпадает ровно одно из dateOfBirth или dateAsAt. Рассмотрим возраст человека, родившегося 1 марта 2003 г. и 29 февраля 2004 г. Чтобы исправить это, вам необходимо провести лексикографическое сравнение пар (Месяц, День-месяц) и использовать его для условного выражения.
он также не будет показывать правильный возраст вашего дня рождения.
Это странный способ сделать это, но если вы отформатируете дату до yyyymmdd и вычтите дату рождения из текущей даты, тогда отбросьте последние 4 цифры вашего возраста :)
Я не знаю C#, но верю, что это будет работать на любом языке.
20080814 - 19800703 = 280111
Отбросьте последние 4 цифры = 28.
Код C#:
int now = int.Parse(DateTime.Now.ToString("yyyyMMdd"));
int dob = int.Parse(dateOfBirth.ToString("yyyyMMdd"));
int age = (now - dob) / 10000;
Или, в качестве альтернативы, без преобразования всех типов в виде метода расширения. Проверка ошибок опущена:
public static Int32 GetAge(this DateTime dateOfBirth)
{
var today = DateTime.Today;
var a = (today.Year * 100 + today.Month) * 100 + today.Day;
var b = (dateOfBirth.Year * 100 + dateOfBirth.Month) * 100 + dateOfBirth.Day;
return (a - b) / 10000;
}
На самом деле это отлично подходит для использования в MS-SQL с полями datetime (общее количество дней с 01-011900)
в альтернативном ответе вы можете избежать целочисленного переполнения, вычтя годы, затем вычтите месяц * 30,5 + день и разделите на 366
@numerek Пожалуйста, опубликуйте свои предлагаемые изменения в качестве собственного ответа. Как бы то ни было, текущий год, умноженный на 10000, далек от целочисленного переполнения на два порядка. 20,150,000 и 2,147,483,648
Было бы лучше добавить параметр: float.Parse (x, CultureInfo.InvariantCulture), потому что некоторые местные жители используют запятую вместо точки для разделителя в дробных числах.
Год DateTime.MaxValue равен 9999, год MinValue равен 1, переполнение невозможно.
В этом ответе предполагается, что у детей високосного дня свои дни рождения 1 марта невисокосных лет.
задаю вопрос по очень старому посту int age = (now - dob) / 10000;, почему вы вычитаете на 10000?
@LongChalk 20180101 - 20171231 = 8870. Отбросьте последние 4 цифры, и вы получите (подразумеваемый) 0 для возраста. Как вы получили 1?
@RufusL Это 0, а не 1. floor(8870 / 10000) == 0. Вы «считаете» десять тысяч, а в 8870 у вас ноль десять тысяч.
Приступить к этому довольно поздно, но вы также можете включить проверку различий TimeOfDay, чтобы получить точный возраст, по крайней мере, на C#.
Я создал функцию, определяемую пользователем SQL Server, чтобы вычислить возраст человека с учетом даты его рождения. Это полезно, когда вам это нужно как часть запроса:
using System;
using System.Data;
using System.Data.Sql;
using System.Data.SqlClient;
using System.Data.SqlTypes;
using Microsoft.SqlServer.Server;
public partial class UserDefinedFunctions
{
[SqlFunction(DataAccess = DataAccessKind.Read)]
public static SqlInt32 CalculateAge(string strBirthDate)
{
DateTime dtBirthDate = new DateTime();
dtBirthDate = Convert.ToDateTime(strBirthDate);
DateTime dtToday = DateTime.Now;
// get the difference in years
int years = dtToday.Year - dtBirthDate.Year;
// subtract another year if we're before the
// birth day in the current year
if (dtToday.Month < dtBirthDate.Month || (dtToday.Month == dtBirthDate.Month && dtToday.Day < dtBirthDate.Day))
years=years-1;
int intCustomerAge = years;
return intCustomerAge;
}
};
Я думаю, что в TimeSpan есть все, что нам нужно, без необходимости прибегать к 365,25 (или любому другому приближению). Расширяя пример Августа:
DateTime myBD = new DateTime(1980, 10, 10);
TimeSpan difference = DateTime.Now.Subtract(myBD);
textBox1.Text = difference.Years + " years " + difference.Months + " Months " + difference.Days + " days";
Неа. TimeSpan как дни, но не месяцы или годы
Мое предложение
int age = (int) ((DateTime.Now - bday).TotalDays/365.242199);
Кажется, что год меняется в нужную дату. (Я прошел тестирование до 107 лет.)
Не думаю, что Гарри Патч оценил бы вашу методологию выборочного тестирования: latimes.com/news/obituaries/…
Google говорит days in a year = 365.242199
Средняя продолжительность года по григорианскому календарю составляет 365,2425 дней.
Я бы сказал, это одно из самых простых решений и это достаточно хорошо. Какая разница, если я за полдня до своего X дня рождения, а программа говорит, что мне X лет. Программа более-менее правильная, хотя и не математически. Мне очень нравится это решение.
^^ Потому что иногда это важно. В моем тестировании это не удается в день рождения человека, он сообщает о том, что он моложе, чем он.
Я бы тоже начал с этого, но при тестировании он часто ошибается с датой рождения человека.
@ChadT, какое решение вы придумали, чтобы оно работало? пользователи постоянно жалуются на то, что День рождения не работает в их дни рождения, и смотрят, как истекают часы.
Не собираюсь тестировать его, но может показаться, что если у вас проблема только в том, что вы хотите, чтобы результат увеличивался на день раньше, вам, вероятно, удастся обойтись: int age = (int) (((DateTime.Now - bday).TotalDays+1)/365.242199);
Его следует проводить выборочные испытания в течение четырех лет подряд, поскольку ошибка накапливается (сбрасывается на один день каждые четыре года к високосному дню - как сегодня!).
Я потратил некоторое время на работу над этим и придумал это, чтобы вычислить чей-то возраст в годах, месяцах и днях. Я тестировал проблему 29 февраля и високосные годы, и, похоже, это сработало, буду признателен за любые отзывы:
public void LoopAge(DateTime myDOB, DateTime FutureDate)
{
int years = 0;
int months = 0;
int days = 0;
DateTime tmpMyDOB = new DateTime(myDOB.Year, myDOB.Month, 1);
DateTime tmpFutureDate = new DateTime(FutureDate.Year, FutureDate.Month, 1);
while (tmpMyDOB.AddYears(years).AddMonths(months) < tmpFutureDate)
{
months++;
if (months > 12)
{
years++;
months = months - 12;
}
}
if (FutureDate.Day >= myDOB.Day)
{
days = days + FutureDate.Day - myDOB.Day;
}
else
{
months--;
if (months < 0)
{
years--;
months = months + 12;
}
days +=
DateTime.DaysInMonth(
FutureDate.AddMonths(-1).Year, FutureDate.AddMonths(-1).Month
) + FutureDate.Day - myDOB.Day;
}
//add an extra day if the dob is a leap day
if (DateTime.IsLeapYear(myDOB.Year) && myDOB.Month == 2 && myDOB.Day == 29)
{
//but only if the future date is less than 1st March
if (FutureDate >= new DateTime(FutureDate.Year, 3, 1))
days++;
}
}
Вот однострочник:
int age = new DateTime(DateTime.Now.Subtract(birthday).Ticks).Year-1;
Сломано. Сделано тестируемым: общедоступный статический int CalculateAge (DateTime dateOfBirth, DateTime dateToCalculateAge) {return new DateTime (dateToCalculateAge.Subtract (dateOfBirth) .Ticks) .Yea r - 1; } ... Дает возраст 14, когда я ввожу 1990-06-01 и вычисляю возраст на день ДО его 14-летия (1990-05-31).
Вот решение.
DateTime dateOfBirth = new DateTime(2000, 4, 18);
DateTime currentDate = DateTime.Now;
int ageInYears = 0;
int ageInMonths = 0;
int ageInDays = 0;
ageInDays = currentDate.Day - dateOfBirth.Day;
ageInMonths = currentDate.Month - dateOfBirth.Month;
ageInYears = currentDate.Year - dateOfBirth.Year;
if (ageInDays < 0)
{
ageInDays += DateTime.DaysInMonth(currentDate.Year, currentDate.Month);
ageInMonths = ageInMonths--;
if (ageInMonths < 0)
{
ageInMonths += 12;
ageInYears--;
}
}
if (ageInMonths < 0)
{
ageInMonths += 12;
ageInYears--;
}
Console.WriteLine("{0}, {1}, {2}", ageInYears, ageInMonths, ageInDays);
Со строкой concat это было бы возможно: 47 лет 11 пн 7 дней
Вот фрагмент теста:
DateTime bDay = new DateTime(2000, 2, 29);
DateTime now = new DateTime(2009, 2, 28);
MessageBox.Show(string.Format("Test {0} {1} {2}",
CalculateAgeWrong1(bDay, now), // outputs 9
CalculateAgeWrong2(bDay, now), // outputs 9
CalculateAgeCorrect(bDay, now), // outputs 8
CalculateAgeCorrect2(bDay, now))); // outputs 8
Вот вам методы:
public int CalculateAgeWrong1(DateTime birthDate, DateTime now)
{
return new DateTime(now.Subtract(birthDate).Ticks).Year - 1;
}
public int CalculateAgeWrong2(DateTime birthDate, DateTime now)
{
int age = now.Year - birthDate.Year;
if (now < birthDate.AddYears(age))
age--;
return age;
}
public int CalculateAgeCorrect(DateTime birthDate, DateTime now)
{
int age = now.Year - birthDate.Year;
if (now.Month < birthDate.Month || (now.Month == birthDate.Month && now.Day < birthDate.Day))
age--;
return age;
}
public int CalculateAgeCorrect2(DateTime birthDate, DateTime now)
{
int age = now.Year - birthDate.Year;
// For leap years we need this
if (birthDate > now.AddYears(-age))
age--;
// Don't use:
// if (birthDate.AddYears(age) > now)
// age--;
return age;
}
Хотя этот код работает, он утверждает, что человек, родившийся в високосный день, достигает следующего года возраста 1 марта в невисокосные годы, а не 28 февраля. На самом деле любой вариант может быть правильным. Википедия может что-то сказать об этом. Так что, хотя ваш код не «неправильный», ни одно решение не является общепринятым.
@MattJohnson Я думаю, что это действительно так. Если мой день рождения был 29 февраля, то 28 февраля мой день рождения не прошел, и я все еще должен быть в том же возрасте, что и 27 февраля. Однако 1 марта мой день рождения закончился, и я должен достичь следующего возраста. В США у предприятия, продающего алкоголь, будет вывеска, на которой написано что-то вроде «Если вы родились после этого дня в ГГГГ, вы не можете покупать алкоголь» (где ГГГГ меняется каждый год). Это означает, что человек, родившийся 29 февраля, не может покупать алкоголь 28 февраля того года, когда ему исполняется 21 год (в большинстве случаев), и поддерживает идею о том, что они не на год старше до 1 марта.
@ jfren484 - прочтите статью в Википедии. Он значительно варьируется в зависимости от юрисдикции.
@ jfren484 Ваше утверждение не имеет ничего общего с философией; но все дело в ваше собственное личное чувство. Когда человек, родившийся 29 февраля, «возраст» не имеет большого значения, если только возраст не образует «возрастную границу» (например, может покупать алкоголь, голосовать, получать пенсию, служить в армии, получать водительские права). Рассмотрим возраст употребления алкоголя в США (21 год): для большинства людей это 7670 дней. Это 7671 день, если они родились до 29 февраля в високосном году или с 1 марта до високосного года. При рождении 29 февраля: 28 февраля - 7670 дней, а 1 марта - 7671 день. Выбор произвольный может идти в любом направлении.
@CraigYoung Вы не понимаете, что я имел в виду с философской точки зрения. Я использовал этот термин как противопоставление юридически. Если кто-то пишет приложение, в котором необходимо знать возраст человека, то все, что ему нужно знать, - это то, как в правовых юрисдикциях, в которых его приложение используется в / для лечения людей, родившихся 29 февраля. Если, однако, мы если говорить о том, как обращаться с этим должен, то это по определению философия. И да, мнение, которое я высказал, является моим собственным мнением, но, как я уже сказал, я думаю, что было бы легче выступать за 1 марта, чем за 28 февраля.
Это может сработать:
public override bool IsValid(DateTime value)
{
_dateOfBirth = value;
var yearsOld = (double) (DateTime.Now.Subtract(_dateOfBirth).TotalDays/365);
if (yearsOld > 18)
return true;
return false;
}
Вау. Почему значение является объектом, а не DateTime? Сигнатура метода должна быть public override bool Is18OrOlder(DateTime birthday). А как быть с людьми, родившимися 29 февраля? Кто сказал, что мы пытаемся проверить, исполнилось ли пользователю 18 лет? Вопрос был в том, "как я могу рассчитать чей-то возраст?"
Как это случилось? Я даже не помню, как ставил IsValid как объект. Это должно быть DateTime!
Почему бы не использовать if вместо заявления return yearsOld > 18;?
Я создал структуру Age, которая выглядит так:
public struct Age : IEquatable<Age>, IComparable<Age>
{
private readonly int _years;
private readonly int _months;
private readonly int _days;
public int Years { get { return _years; } }
public int Months { get { return _months; } }
public int Days { get { return _days; } }
public Age( int years, int months, int days ) : this()
{
_years = years;
_months = months;
_days = days;
}
public static Age CalculateAge( DateTime dateOfBirth, DateTime date )
{
// Here is some logic that ressembles Mike's solution, although it
// also takes into account months & days.
// Ommitted for brevity.
return new Age (years, months, days);
}
// Ommited Equality, Comparable, GetHashCode, functionality for brevity.
}
Вот небольшой пример кода для C#, который я придумал, будьте осторожны с крайними случаями, особенно високосными годами, не все вышеперечисленные решения учитывают их. Выдвижение ответа как DateTime может вызвать проблемы, поскольку вы можете в конечном итоге попытаться поместить слишком много дней в конкретный месяц, например. 30 дней в февр.
public string LoopAge(DateTime myDOB, DateTime FutureDate)
{
int years = 0;
int months = 0;
int days = 0;
DateTime tmpMyDOB = new DateTime(myDOB.Year, myDOB.Month, 1);
DateTime tmpFutureDate = new DateTime(FutureDate.Year, FutureDate.Month, 1);
while (tmpMyDOB.AddYears(years).AddMonths(months) < tmpFutureDate)
{
months++;
if (months > 12)
{
years++;
months = months - 12;
}
}
if (FutureDate.Day >= myDOB.Day)
{
days = days + FutureDate.Day - myDOB.Day;
}
else
{
months--;
if (months < 0)
{
years--;
months = months + 12;
}
days = days + (DateTime.DaysInMonth(FutureDate.AddMonths(-1).Year, FutureDate.AddMonths(-1).Month) + FutureDate.Day) - myDOB.Day;
}
//add an extra day if the dob is a leap day
if (DateTime.IsLeapYear(myDOB.Year) && myDOB.Month == 2 && myDOB.Day == 29)
{
//but only if the future date is less than 1st March
if (FutureDate >= new DateTime(FutureDate.Year, 3,1))
days++;
}
return "Years: " + years + " Months: " + months + " Days: " + days;
}
Мне больше всего нравится это решение, однако при подсчете месяцев оно должно быть if (months> = 12). Попробуйте 6-8-2012 - 6-4-1993 для проверки.
Я использую это:
public static class DateTimeExtensions
{
public static int Age(this DateTime birthDate)
{
return Age(birthDate, DateTime.Now);
}
public static int Age(this DateTime birthDate, DateTime offsetDate)
{
int result=0;
result = offsetDate.Year - birthDate.Year;
if (offsetDate.DayOfYear < birthDate.DayOfYear)
{
result--;
}
return result;
}
}
Делать это просто (и, возможно, глупо :)).
DateTime birth = new DateTime(1975, 09, 27, 01, 00, 00, 00);
TimeSpan ts = DateTime.Now - birth;
Console.WriteLine("You are approximately " + ts.TotalSeconds.ToString() + " seconds old.");
TimeSpan был моим первым выбором, но обнаружил, что он не предлагает свойство TotalYears. Вы можете попробовать (ts.TotalDays / 365), но он не учитывает високосные годы и т. д.
private int GetAge(int _year, int _month, int _day
{
DateTime yourBirthDate= new DateTime(_year, _month, _day);
DateTime todaysDateTime = DateTime.Today;
int noOfYears = todaysDateTime.Year - yourBirthDate.Year;
if (DateTime.Now.Month < yourBirthDate.Month ||
(DateTime.Now.Month == yourBirthDate.Month && DateTime.Now.Day < yourBirthDate.Day))
{
noOfYears--;
}
return noOfYears;
}
Самый простой способ, который я когда-либо находил, - это. Он правильно работает для регионов США и Западной Европы. Не могу разговаривать с другими местами, особенно с такими местами, как Китай. Не более 4 дополнительных сравнений после первоначального вычисления возраста.
public int AgeInYears(DateTime birthDate, DateTime referenceDate)
{
Debug.Assert(referenceDate >= birthDate,
"birth date must be on or prior to the reference date");
DateTime birth = birthDate.Date;
DateTime reference = referenceDate.Date;
int years = (reference.Year - birth.Year);
//
// an offset of -1 is applied if the birth date has
// not yet occurred in the current year.
//
if (reference.Month > birth.Month);
else if (reference.Month < birth.Month)
--years;
else // in birth month
{
if (reference.Day < birth.Day)
--years;
}
return years ;
}
Я просматривал ответы на этот вопрос и заметил, что никто не упомянул нормативные / правовые последствия високосных родов. Например, согласно Википедии, если вы родились 29 февраля в разных юрисдикциях, ваш день рождения не в високосный год может быть разным:
И, насколько я могу судить, в США законодательные акты по этому поводу ничего не говорят, оставляя это на усмотрение общего права и того, как различные регулирующие органы определяют вещи в своих правилах.
С этой целью улучшение:
public enum LeapDayRule
{
OrdinalDay = 1 ,
LastDayOfMonth = 2 ,
}
static int ComputeAgeInYears(DateTime birth, DateTime reference, LeapYearBirthdayRule ruleInEffect)
{
bool isLeapYearBirthday = CultureInfo.CurrentCulture.Calendar.IsLeapDay(birth.Year, birth.Month, birth.Day);
DateTime cutoff;
if (isLeapYearBirthday && !DateTime.IsLeapYear(reference.Year))
{
switch (ruleInEffect)
{
case LeapDayRule.OrdinalDay:
cutoff = new DateTime(reference.Year, 1, 1)
.AddDays(birth.DayOfYear - 1);
break;
case LeapDayRule.LastDayOfMonth:
cutoff = new DateTime(reference.Year, birth.Month, 1)
.AddMonths(1)
.AddDays(-1);
break;
default:
throw new InvalidOperationException();
}
}
else
{
cutoff = new DateTime(reference.Year, birth.Month, birth.Day);
}
int age = (reference.Year - birth.Year) + (reference >= cutoff ? 0 : -1);
return age < 0 ? 0 : age;
}
Следует отметить, что этот код предполагает:
Простой ответ на этот вопрос - применить AddYears, как показано ниже, потому что это единственный собственный метод добавления лет к 29 февраля високосных лет и получения правильного результата 28 февраля для обычных лет.
Некоторые считают, что 1 марта - день рождения прыгунов, но ни .Net, ни какие-либо официальные правила не поддерживают этого, и не существует общепринятой логики, объясняющей, почему у некоторых рожденных в феврале 75% дней рождения должны быть в другом месяце.
Кроме того, метод Age может быть добавлен как расширение DateTime. Таким образом можно получить возраст самым простым способом:
int age = BirthDate.Age ();
public static class DateTimeExtensions
{
/// <summary>
/// Calculates the age in years of the current System.DateTime object today.
/// </summary>
/// <param name = "birthDate">The date of birth</param>
/// <returns>Age in years today. 0 is returned for a future date of birth.</returns>
public static int Age(this DateTime birthDate)
{
return Age(birthDate, DateTime.Today);
}
/// <summary>
/// Calculates the age in years of the current System.DateTime object on a later date.
/// </summary>
/// <param name = "birthDate">The date of birth</param>
/// <param name = "laterDate">The date on which to calculate the age.</param>
/// <returns>Age in years on a later day. 0 is returned as minimum.</returns>
public static int Age(this DateTime birthDate, DateTime laterDate)
{
int age;
age = laterDate.Year - birthDate.Year;
if (age > 0)
{
age -= Convert.ToInt32(laterDate.Date < birthDate.Date.AddYears(age));
}
else
{
age = 0;
}
return age;
}
}
Теперь запустим этот тест:
class Program
{
static void Main(string[] args)
{
RunTest();
}
private static void RunTest()
{
DateTime birthDate = new DateTime(2000, 2, 28);
DateTime laterDate = new DateTime(2011, 2, 27);
string iso = "yyyy-MM-dd";
for (int i = 0; i < 3; i++)
{
for (int j = 0; j < 3; j++)
{
Console.WriteLine("Birth date: " + birthDate.AddDays(i).ToString(iso) + " Later date: " + laterDate.AddDays(j).ToString(iso) + " Age: " + birthDate.AddDays(i).Age(laterDate.AddDays(j)).ToString());
}
}
Console.ReadKey();
}
}
Вот пример критической даты:
Дата рождения: 29.02.2000 Позднее: 28.02.2011 Возраст: 11
Выход:
{
Birth date: 2000-02-28 Later date: 2011-02-27 Age: 10
Birth date: 2000-02-28 Later date: 2011-02-28 Age: 11
Birth date: 2000-02-28 Later date: 2011-03-01 Age: 11
Birth date: 2000-02-29 Later date: 2011-02-27 Age: 10
Birth date: 2000-02-29 Later date: 2011-02-28 Age: 11
Birth date: 2000-02-29 Later date: 2011-03-01 Age: 11
Birth date: 2000-03-01 Later date: 2011-02-27 Age: 10
Birth date: 2000-03-01 Later date: 2011-02-28 Age: 10
Birth date: 2000-03-01 Later date: 2011-03-01 Age: 11
}
И для более поздней даты 2012-02-28:
{
Birth date: 2000-02-28 Later date: 2012-02-28 Age: 12
Birth date: 2000-02-28 Later date: 2012-02-29 Age: 12
Birth date: 2000-02-28 Later date: 2012-03-01 Age: 12
Birth date: 2000-02-29 Later date: 2012-02-28 Age: 11
Birth date: 2000-02-29 Later date: 2012-02-29 Age: 12
Birth date: 2000-02-29 Later date: 2012-03-01 Age: 12
Birth date: 2000-03-01 Later date: 2012-02-28 Age: 11
Birth date: 2000-03-01 Later date: 2012-02-29 Age: 11
Birth date: 2000-03-01 Later date: 2012-03-01 Age: 12
}
Комментарий относительно того, что 29 февраля день рождения должен быть 1 марта, технически, иметь его 28-го слишком рано (фактически, на 1 день раньше). 1-го уже на один день поздно. Но поскольку день рождения находится между ними, использование 1-го числа для расчета возраста в невисокосных годах для меня имеет больше смысла, поскольку этот человек действительно становится таким же старым 1 (и 2 и 3) марта каждого года, но не 28 февраля.
С точки зрения разработки программного обеспечения, писать это как метод расширения для меня не имеет большого смысла. date.Age(other)?
Как насчет этого решения?
static string CalcAge(DateTime birthDay)
{
DateTime currentDate = DateTime.Now;
int approximateAge = currentDate.Year - birthDay.Year;
int daysToNextBirthDay = (birthDay.Month * 30 + birthDay.Day) -
(currentDate.Month * 30 + currentDate.Day) ;
if (approximateAge == 0 || approximateAge == 1)
{
int month = Math.Abs(daysToNextBirthDay / 30);
int days = Math.Abs(daysToNextBirthDay % 30);
if (month == 0)
return "Your age is: " + daysToNextBirthDay + " days";
return "Your age is: " + month + " months and " + days + " days"; ;
}
if (daysToNextBirthDay > 0)
return "Your age is: " + --approximateAge + " Years";
return "Your age is: " + approximateAge + " Years"; ;
}
2 Основные проблемы, которые необходимо решить:
1. Рассчитайте точный возраст - в годах, месяцах, днях и т. д.
2. Рассчитайте общепринятый возраст. - людям обычно все равно, сколько им лет, им просто важно, когда у них день рождения в текущем году.
Решение для 1 очевидно:
DateTime birth = DateTime.Parse("1.1.2000");
DateTime today = DateTime.Today; //we usually don't care about birth time
TimeSpan age = today - birth; //.NET FCL should guarantee this as precise
double ageInDays = age.TotalDays; //total number of days ... also precise
double daysInYear = 365.2425; //statistical value for 400 years
double ageInYears = ageInDays / daysInYear; //can be shifted ... not so precise
Решение для 2 - это решение, которое не так точно определяет общий возраст, но воспринимается людьми как точное. Люди также обычно используют его, когда вычисляют свой возраст «вручную»:
DateTime birth = DateTime.Parse("1.1.2000");
DateTime today = DateTime.Today;
int age = today.Year - birth.Year; //people perceive their age in years
if (today.Month < birth.Month ||
((today.Month == birth.Month) && (today.Day < birth.Day)))
{
age--; //birthday in current year not yet reached, we are 1 year younger ;)
//+ no birthday for 29.2. guys ... sorry, just wrong date for birth
}
Примечания к 2 .:
Еще одно замечание ... Я бы создал для него 2 статических перегруженных метода, один для универсального использования, второй для удобства использования:
public static int GetAge(DateTime bithDay, DateTime today)
{
//chosen solution method body
}
public static int GetAge(DateTime birthDay)
{
return GetAge(birthDay, DateTime.Now);
}
Следующий подход (выдержка из Библиотека периодов времени для .NET класса DateDiff) рассматривает календарь информации о культуре:
// ----------------------------------------------------------------------
private static int YearDiff( DateTime date1, DateTime date2 )
{
return YearDiff( date1, date2, DateTimeFormatInfo.CurrentInfo.Calendar );
} // YearDiff
// ----------------------------------------------------------------------
private static int YearDiff( DateTime date1, DateTime date2, Calendar calendar )
{
if ( date1.Equals( date2 ) )
{
return 0;
}
int year1 = calendar.GetYear( date1 );
int month1 = calendar.GetMonth( date1 );
int year2 = calendar.GetYear( date2 );
int month2 = calendar.GetMonth( date2 );
// find the the day to compare
int compareDay = date2.Day;
int compareDaysPerMonth = calendar.GetDaysInMonth( year1, month1 );
if ( compareDay > compareDaysPerMonth )
{
compareDay = compareDaysPerMonth;
}
// build the compare date
DateTime compareDate = new DateTime( year1, month2, compareDay,
date2.Hour, date2.Minute, date2.Second, date2.Millisecond );
if ( date2 > date1 )
{
if ( compareDate < date1 )
{
compareDate = compareDate.AddYears( 1 );
}
}
else
{
if ( compareDate > date1 )
{
compareDate = compareDate.AddYears( -1 );
}
}
return year2 - calendar.GetYear( compareDate );
} // YearDiff
Использование:
// ----------------------------------------------------------------------
public void CalculateAgeSamples()
{
PrintAge( new DateTime( 2000, 02, 29 ), new DateTime( 2009, 02, 28 ) );
// > Birthdate=29.02.2000, Age at 28.02.2009 is 8 years
PrintAge( new DateTime( 2000, 02, 29 ), new DateTime( 2012, 02, 28 ) );
// > Birthdate=29.02.2000, Age at 28.02.2012 is 11 years
} // CalculateAgeSamples
// ----------------------------------------------------------------------
public void PrintAge( DateTime birthDate, DateTime moment )
{
Console.WriteLine( "Birthdate = {0:d}, Age at {1:d} is {2} years", birthDate, moment, YearDiff( birthDate, moment ) );
} // PrintAge
Вот расширитель DateTime, который добавляет вычисление возраста к объекту DateTime.
public static class AgeExtender
{
public static int GetAge(this DateTime dt)
{
int d = int.Parse(dt.ToString("yyyyMMdd"));
int t = int.Parse(DateTime.Today.ToString("yyyyMMdd"));
return (t-d)/10000;
}
}
тьфу, не делай этого. ToString и int.Parse относительно дороги, и, хотя я против микрооптимизации, скрывать дорогостоящие функции в методах расширения, которые должны быть тривиальными операциями, не является хорошей идеей.
Кроме того, это дубликат ответа ScArcher2: stackoverflow.com/questions/9/…
Яур, мне очень нравится решение Элмера, основанное на DayOfYear, вероятно, более эффективное, чем мое. Обратите внимание, что моей целью не было изменить алгоритм ScArcher2, я чувствовал, что это было бы грубо. Это было просто, чтобы показать, как реализовать метод расширения.
Я сделал одно небольшое изменение в ответе Марка Соена: я переписал третью строку, чтобы выражение можно было легче анализировать.
public int AgeInYears(DateTime bday)
{
DateTime now = DateTime.Today;
int age = now.Year - bday.Year;
if (bday.AddYears(age) > now)
age--;
return age;
}
Я также превратил его в функцию для ясности.
Я использовал решение ScArcher2 для точного расчета возраста людей, но мне нужно было пойти дальше и вычислить их месяцы и дни вместе с годами.
public static Dictionary<string,int> CurrentAgeInYearsMonthsDays(DateTime? ndtBirthDate, DateTime? ndtReferralDate)
{
//----------------------------------------------------------------------
// Can't determine age if we don't have a dates.
//----------------------------------------------------------------------
if (ndtBirthDate == null) return null;
if (ndtReferralDate == null) return null;
DateTime dtBirthDate = Convert.ToDateTime(ndtBirthDate);
DateTime dtReferralDate = Convert.ToDateTime(ndtReferralDate);
//----------------------------------------------------------------------
// Create our Variables
//----------------------------------------------------------------------
Dictionary<string, int> dYMD = new Dictionary<string,int>();
int iNowDate, iBirthDate, iYears, iMonths, iDays;
string sDif = "";
//----------------------------------------------------------------------
// Store off current date/time and DOB into local variables
//----------------------------------------------------------------------
iNowDate = int.Parse(dtReferralDate.ToString("yyyyMMdd"));
iBirthDate = int.Parse(dtBirthDate.ToString("yyyyMMdd"));
//----------------------------------------------------------------------
// Calculate Years
//----------------------------------------------------------------------
sDif = (iNowDate - iBirthDate).ToString();
iYears = int.Parse(sDif.Substring(0, sDif.Length - 4));
//----------------------------------------------------------------------
// Store Years in Return Value
//----------------------------------------------------------------------
dYMD.Add("Years", iYears);
//----------------------------------------------------------------------
// Calculate Months
//----------------------------------------------------------------------
if (dtBirthDate.Month > dtReferralDate.Month)
iMonths = 12 - dtBirthDate.Month + dtReferralDate.Month - 1;
else
iMonths = dtBirthDate.Month - dtReferralDate.Month;
//----------------------------------------------------------------------
// Store Months in Return Value
//----------------------------------------------------------------------
dYMD.Add("Months", iMonths);
//----------------------------------------------------------------------
// Calculate Remaining Days
//----------------------------------------------------------------------
if (dtBirthDate.Day > dtReferralDate.Day)
//Logic: Figure out the days in month previous to the current month, or the admitted month.
// Subtract the birthday from the total days which will give us how many days the person has lived since their birthdate day the previous month.
// then take the referral date and simply add the number of days the person has lived this month.
//If referral date is january, we need to go back to the following year's December to get the days in that month.
if (dtReferralDate.Month == 1)
iDays = DateTime.DaysInMonth(dtReferralDate.Year - 1, 12) - dtBirthDate.Day + dtReferralDate.Day;
else
iDays = DateTime.DaysInMonth(dtReferralDate.Year, dtReferralDate.Month - 1) - dtBirthDate.Day + dtReferralDate.Day;
else
iDays = dtReferralDate.Day - dtBirthDate.Day;
//----------------------------------------------------------------------
// Store Days in Return Value
//----------------------------------------------------------------------
dYMD.Add("Days", iDays);
return dYMD;
}
Я хочу добавить вычисления календаря на иврите (или другой календарь System.Globalization можно использовать таким же образом), используя переписанные функции из этого потока:
Public Shared Function CalculateAge(BirthDate As DateTime) As Integer
Dim HebCal As New System.Globalization.HebrewCalendar ()
Dim now = DateTime.Now()
Dim iAge = HebCal.GetYear(now) - HebCal.GetYear(BirthDate)
Dim iNowMonth = HebCal.GetMonth(now), iBirthMonth = HebCal.GetMonth(BirthDate)
If iNowMonth < iBirthMonth Or (iNowMonth = iBirthMonth AndAlso HebCal.GetDayOfMonth(now) < HebCal.GetDayOfMonth(BirthDate)) Then iAge -= 1
Return iAge
End Function
Вопрос: «Как в C# рассчитать чей-то возраст ...».
C# и vb одинаковы для этой идеи.
Это просто и кажется мне точным. Я делаю предположение в целях високосных лет, что независимо от того, когда человек решает отметить день рождения, технически он не старше на год, пока с его последнего дня рождения не пройдет 365 дней (т.е. 28 февраля не делает его на год старше) .
DateTime now = DateTime.Today;
DateTime birthday = new DateTime(1991, 02, 03);//3rd feb
int age = now.Year - birthday.Year;
if (now.Month < birthday.Month || (now.Month == birthday.Month && now.Day < birthday.Day))//not had bday this year yet
age--;
return age;
Попробуйте это решение, оно работает.
int age = (Int32.Parse(DateTime.Today.ToString("yyyyMMdd")) -
Int32.Parse(birthday.ToString("yyyyMMdd rawrrr"))) / 10000;
Это не прямой ответ, а скорее философское рассуждение о рассматриваемой проблеме с квазинаучной точки зрения.
Я бы сказал, что вопрос не определяет единицу измерения или культуру, в которой измеряется возраст, большинство ответов, похоже, предполагают целочисленное годовое представление. Единицей СИ для времени является second, следовательно, правильный общий ответ должен быть (конечно, принимая нормализованный DateTime и не принимая во внимание релятивистские эффекты):
var lifeInSeconds = (DateTime.Now.Ticks - then.Ticks)/TickFactor;
Согласно христианскому способу исчисления возраста в годах:
var then = ... // Then, in this case the birthday
var now = DateTime.UtcNow;
int age = now.Year - then.Year;
if (now.AddYears(-age) < then) age--;
В финансах есть аналогичная проблема при вычислении чего-то, что часто называют Доля подсчета дней, что примерно соответствует количеству лет для данного периода. И проблема возраста - это действительно проблема измерения времени.
Пример условного обозначения фактического / фактического (считая все дни "правильно"):
DateTime start, end = .... // Whatever, assume start is before end
double startYearContribution = 1 - (double) start.DayOfYear / (double) (DateTime.IsLeapYear(start.Year) ? 366 : 365);
double endYearContribution = (double)end.DayOfYear / (double)(DateTime.IsLeapYear(end.Year) ? 366 : 365);
double middleContribution = (double) (end.Year - start.Year - 1);
double DCF = startYearContribution + endYearContribution + middleContribution;
Другой довольно распространенный способ измерения времени - "сериализация" (чувак, назвавший это соглашение о дате, должно быть, серьезно споткнулся):
DateTime start, end = .... // Whatever, assume start is before end
int days = (end - start).Days;
Интересно, сколько времени нам нужно пройти до того, как релятивистская эпоха в секундах станет более полезной, чем грубое приближение циклов Земля вокруг Солнца в течение всей жизни человека :) Или, другими словами, когда период должен быть определен местом или функция, представляющая движение, чтобы быть действительной :)
Что такое TickFactor?
@Protiguous Ticks в секунду, используется для нормализации тиков до секунд.
Нужно ли учитывать людей младше 1 года? В китайской культуре мы описываем возраст маленьких детей как 2 месяца или 4 недели.
Ниже представлена моя реализация, она не так проста, как я себе представлял, особенно с датой вроде 28 февраля.
public static string HowOld(DateTime birthday, DateTime now)
{
if (now < birthday)
throw new ArgumentOutOfRangeException("birthday must be less than now.");
TimeSpan diff = now - birthday;
int diffDays = (int)diff.TotalDays;
if (diffDays > 7)//year, month and week
{
int age = now.Year - birthday.Year;
if (birthday > now.AddYears(-age))
age--;
if (age > 0)
{
return age + (age > 1 ? " years" : " year");
}
else
{// month and week
DateTime d = birthday;
int diffMonth = 1;
while (d.AddMonths(diffMonth) <= now)
{
diffMonth++;
}
age = diffMonth-1;
if (age == 1 && d.Day > now.Day)
age--;
if (age > 0)
{
return age + (age > 1 ? " months" : " month");
}
else
{
age = diffDays / 7;
return age + (age > 1 ? " weeks" : " week");
}
}
}
else if (diffDays > 0)
{
int age = diffDays;
return age + (age > 1 ? " days" : " day");
}
else
{
int age = diffDays;
return "just born";
}
}
Эта реализация прошла тестовые примеры ниже.
[TestMethod]
public void TestAge()
{
string age = HowOld(new DateTime(2011, 1, 1), new DateTime(2012, 11, 30));
Assert.AreEqual("1 year", age);
age = HowOld(new DateTime(2011, 11, 30), new DateTime(2012, 11, 30));
Assert.AreEqual("1 year", age);
age = HowOld(new DateTime(2001, 1, 1), new DateTime(2012, 11, 30));
Assert.AreEqual("11 years", age);
age = HowOld(new DateTime(2012, 1, 1), new DateTime(2012, 11, 30));
Assert.AreEqual("10 months", age);
age = HowOld(new DateTime(2011, 12, 1), new DateTime(2012, 11, 30));
Assert.AreEqual("11 months", age);
age = HowOld(new DateTime(2012, 10, 1), new DateTime(2012, 11, 30));
Assert.AreEqual("1 month", age);
age = HowOld(new DateTime(2008, 2, 28), new DateTime(2009, 2, 28));
Assert.AreEqual("1 year", age);
age = HowOld(new DateTime(2008, 3, 28), new DateTime(2009, 2, 28));
Assert.AreEqual("11 months", age);
age = HowOld(new DateTime(2008, 3, 28), new DateTime(2009, 3, 28));
Assert.AreEqual("1 year", age);
age = HowOld(new DateTime(2009, 1, 28), new DateTime(2009, 2, 28));
Assert.AreEqual("1 month", age);
age = HowOld(new DateTime(2009, 2, 1), new DateTime(2009, 3, 1));
Assert.AreEqual("1 month", age);
// NOTE.
// new DateTime(2008, 1, 31).AddMonths(1) == new DateTime(2009, 2, 28);
// new DateTime(2008, 1, 28).AddMonths(1) == new DateTime(2009, 2, 28);
age = HowOld(new DateTime(2009, 1, 31), new DateTime(2009, 2, 28));
Assert.AreEqual("4 weeks", age);
age = HowOld(new DateTime(2009, 2, 1), new DateTime(2009, 2, 28));
Assert.AreEqual("3 weeks", age);
age = HowOld(new DateTime(2009, 2, 1), new DateTime(2009, 3, 1));
Assert.AreEqual("1 month", age);
age = HowOld(new DateTime(2012, 11, 5), new DateTime(2012, 11, 30));
Assert.AreEqual("3 weeks", age);
age = HowOld(new DateTime(2012, 11, 1), new DateTime(2012, 11, 30));
Assert.AreEqual("4 weeks", age);
age = HowOld(new DateTime(2012, 11, 20), new DateTime(2012, 11, 30));
Assert.AreEqual("1 week", age);
age = HowOld(new DateTime(2012, 11, 25), new DateTime(2012, 11, 30));
Assert.AreEqual("5 days", age);
age = HowOld(new DateTime(2012, 11, 29), new DateTime(2012, 11, 30));
Assert.AreEqual("1 day", age);
age = HowOld(new DateTime(2012, 11, 30), new DateTime(2012, 11, 30));
Assert.AreEqual("just born", age);
age = HowOld(new DateTime(2000, 2, 29), new DateTime(2009, 2, 28));
Assert.AreEqual("8 years", age);
age = HowOld(new DateTime(2000, 2, 29), new DateTime(2009, 3, 1));
Assert.AreEqual("9 years", age);
Exception e = null;
try
{
age = HowOld(new DateTime(2012, 12, 1), new DateTime(2012, 11, 30));
}
catch (ArgumentOutOfRangeException ex)
{
e = ex;
}
Assert.IsTrue(e != null);
}
Надеюсь, это поможет.
Вот очень простой и понятный пример.
private int CalculateAge()
{
//get birthdate
DateTime dtBirth = Convert.ToDateTime(BirthDatePicker.Value);
int byear = dtBirth.Year;
int bmonth = dtBirth.Month;
int bday = dtBirth.Day;
DateTime dtToday = DateTime.Now;
int tYear = dtToday.Year;
int tmonth = dtToday.Month;
int tday = dtToday.Day;
int age = tYear - byear;
if (bmonth < tmonth)
age--;
else if (bmonth == tmonth && bday>tday)
{
age--;
}
return age;
}
Вот еще один ответ:
public static int AgeInYears(DateTime birthday, DateTime today)
{
return ((today.Year - birthday.Year) * 372 + (today.Month - birthday.Month) * 31 + (today.Day - birthday.Day)) / 372;
}
Это было тщательно протестировано. Это действительно выглядит немного "волшебно". Число 372 - это количество дней в году, если бы каждый месяц состоял из 31 дня.
Объяснение того, почему это работает (снят отсюда):
Let's set
Yn = DateTime.Now.Year, Yb = birthday.Year, Mn = DateTime.Now.Month, Mb = birthday.Month, Dn = DateTime.Now.Day, Db = birthday.Day
age = Yn - Yb + (31*(Mn - Mb) + (Dn - Db)) / 372We know that what we need is either
Yn-Ybif the date has already been reached,Yn-Yb-1if it has not.a) If
Mn<Mb, we have-341 <= 31*(Mn-Mb) <= -31 and -30 <= Dn-Db <= 30
-371 <= 31*(Mn - Mb) + (Dn - Db) <= -1With integer division
(31*(Mn - Mb) + (Dn - Db)) / 372 = -1b) If
Mn=MbandDn<Db, we have31*(Mn - Mb) = 0 and -30 <= Dn-Db <= -1With integer division, again
(31*(Mn - Mb) + (Dn - Db)) / 372 = -1c) If
Mn>Mb, we have31 <= 31*(Mn-Mb) <= 341 and -30 <= Dn-Db <= 30
1 <= 31*(Mn - Mb) + (Dn - Db) <= 371With integer division
(31*(Mn - Mb) + (Dn - Db)) / 372 = 0d) If
Mn=MbandDn>Db, we have31*(Mn - Mb) = 0 and 1 <= Dn-Db <= 30With integer division, again
(31*(Mn - Mb) + (Dn - Db)) / 372 = 0e) If
Mn=MbandDn=Db, we have31*(Mn - Mb) + Dn-Db = 0and therefore
(31*(Mn - Mb) + (Dn - Db)) / 372 = 0
С меньшим количеством конверсий и UtcNow этот код может позаботиться о людях, родившихся 29 февраля високосного года:
public int GetAge(DateTime DateOfBirth)
{
var Now = DateTime.UtcNow;
return Now.Year - DateOfBirth.Year -
(
(
Now.Month > DateOfBirth.Month ||
(Now.Month == DateOfBirth.Month && Now.Day >= DateOfBirth.Day)
) ? 0 : 1
);
}
Это сегодня! (Следующий - через четыре года.)
How come the MSDN help did not tell you that? It looks so obvious:
System.DateTime birthTime = AskTheUser(myUser); // :-)
System.DateTime now = System.DateTime.Now;
System.TimeSpan age = now - birthTime; // As simple as that
double ageInDays = age.TotalDays; // Will you convert to whatever you want yourself?
Что ж, это фантастика, если вы находитесь в одной из нулевых стран на Земле, где возраст взрослых людей измеряется днями.
TimeSpan diff = DateTime.Now - birthdayDateTime;
string age = String.Format("{0:%y} years, {0:%M} months, {0:%d}, days old", diff);
Я не уверен, как именно вы хотите, чтобы он вернулся к вам, поэтому я просто сделал читаемую строку.
Это дает "более подробную информацию" по этому вопросу. Может это то, что ты ищешь
DateTime birth = new DateTime(1974, 8, 29);
DateTime today = DateTime.Now;
TimeSpan span = today - birth;
DateTime age = DateTime.MinValue + span;
// Make adjustment due to MinValue equalling 1/1/1
int years = age.Year - 1;
int months = age.Month - 1;
int days = age.Day - 1;
// Print out not only how many years old they are but give months and days as well
Console.Write("{0} years, {1} months, {2} days", years, months, days);
Это не всегда работает. Добавление Span в DateTime.MinValue может работать, так как это не учитывает високосные годы и т. д. Если вы добавите годы, месяцы и дни к Age с помощью функций AddYears (), AddMonths и AddDays (), он не всегда будет возвращать Datetime. .Настоящая дата.
Сам временной интервал автоматически учитывает високосные годы между двумя датами, поэтому я не уверен, о чем вы. Я спросил на форумах Microsoft, и Microsoft подтвердила, что учитывает високосные годы между двумя датами.
Рассмотрим следующие ДВА сенария. 1st DateTime.Now - 01.01.2001, а ребенок родился 01.01.2000. 2000 год является високосным, и результатом будет 1 год, 0 месяцев и 1 день. Во втором значении DateTime.Now - 01.01.2002, а ребенок родился 01.01.2001. В этом случае результатом будет 1 год, 0 месяцев и 0 дней. Это произойдет, потому что вы добавляете временной интервал в невисокосный год. Если DateTime.MinValue был високосным годом, то результаты были бы 1 год в первый и 0 лет 11 месяцев и 30 дней. (Попробуйте в своем коде).
Голосовать за! Я придумал решение, которое практически идентично (я использовал DateTime.MinValue.AddTicks (span.Ticks) вместо +, но результат тот же, и у вас код на несколько символов меньше).
@AthanasiosKataras, я не понимаю ваш комментарий «Если бы DateTime.MinValue был високосным годом ...». Когда DateTime.MinValue будет високосным годом? Это всегда будет «01.01.10001», что не является високосным годом.
Вы совершенно правы, это не так. Но если бы это было так, то был бы результат. Почему это имеет значение? Это не так. В любом случае прыжок или нет, тогда есть примеры, когда это не работает. Это то, что я хотел показать. DIFF правильный. Пролет учитывает високосные годы. Но ДОБАВЛЕНИЯ к базовой дате - нет. Попробуйте примеры в коде, и вы убедитесь, что я прав.
Чтобы рассчитать возраст с ближайшим возрастом:
var ts = DateTime.Now - new DateTime(1988, 3, 19);
var age = Math.Round(ts.Days / 365.0);
не обязательно верно. Думаю, правильнее было бы разделить на 365,25, чтобы как-то учесть високосные годы.
Этот классический вопрос заслуживает решения Время Нода.
static int GetAge(LocalDate dateOfBirth)
{
Instant now = SystemClock.Instance.Now;
// The target time zone is important.
// It should align with the *current physical location* of the person
// you are talking about. When the whereabouts of that person are unknown,
// then you use the time zone of the person who is *asking* for the age.
// The time zone of birth is irrelevant!
DateTimeZone zone = DateTimeZoneProviders.Tzdb["America/New_York"];
LocalDate today = now.InZone(zone).Date;
Period period = Period.Between(dateOfBirth, today, PeriodUnits.Years);
return (int) period.Years;
}
Использование:
LocalDate dateOfBirth = new LocalDate(1976, 8, 27);
int age = GetAge(dateOfBirth);
Вас также могут заинтересовать следующие улучшения:
Передача часов как IClock вместо использования SystemClock.Instance улучшит тестируемость.
Целевой часовой пояс, скорее всего, изменится, поэтому вам также понадобится параметр DateTimeZone.
См. Также мое сообщение в блоге на эту тему: Дни рождения и другие юбилеи
У меня есть индивидуальный метод расчета возраста, а также сообщение о подтверждении бонуса на случай, если это поможет:
public void GetAge(DateTime dob, DateTime now, out int years, out int months, out int days)
{
years = 0;
months = 0;
days = 0;
DateTime tmpdob = new DateTime(dob.Year, dob.Month, 1);
DateTime tmpnow = new DateTime(now.Year, now.Month, 1);
while (tmpdob.AddYears(years).AddMonths(months) < tmpnow)
{
months++;
if (months > 12)
{
years++;
months = months - 12;
}
}
if (now.Day >= dob.Day)
days = days + now.Day - dob.Day;
else
{
months--;
if (months < 0)
{
years--;
months = months + 12;
}
days += DateTime.DaysInMonth(now.AddMonths(-1).Year, now.AddMonths(-1).Month) + now.Day - dob.Day;
}
if (DateTime.IsLeapYear(dob.Year) && dob.Month == 2 && dob.Day == 29 && now >= new DateTime(now.Year, 3, 1))
days++;
}
private string ValidateDate(DateTime dob) //This method will validate the date
{
int Years = 0; int Months = 0; int Days = 0;
GetAge(dob, DateTime.Now, out Years, out Months, out Days);
if (Years < 18)
message = Years + " is too young. Please try again on your 18th birthday.";
else if (Years >= 65)
message = Years + " is too old. Date of Birth must not be 65 or older.";
else
return null; //Denotes validation passed
}
Вызов метода здесь и передача значения datetime (MM / dd / yyyy, если сервер установлен на локаль США). Замените это любым отображаемым окном сообщения или любым контейнером:
DateTime dob = DateTime.Parse("03/10/1982");
string message = ValidateDate(dob);
lbldatemessage.Visible = !StringIsNullOrWhitespace(message);
lbldatemessage.Text = message ?? ""; //Ternary if message is null then default to empty string
Помните, что вы можете форматировать сообщение как хотите.
Однострочный ответ:
DateTime dateOfBirth = Convert.ToDateTime("01/16/1990");
var age = ((DateTime.Now - dateOfBirth).Days) / 365;
А как насчет високосных лет?
Это один из наиболее точных ответов, который позволяет определить дату рождения 29 февраля по сравнению с любым годом 28 февраля.
public int GetAge(DateTime birthDate)
{
int age = DateTime.Now.Year - birthDate.Year;
if (birthDate.DayOfYear > DateTime.Now.DayOfYear)
age--;
return age;
}
Это сегодня! (Следующий - через четыре года.)
Вместо этого вы можете использовать DateTime.Today, поскольку время не имеет значения для расчета
Просто потому, что я не думаю, что главный ответ так однозначен:
public static int GetAgeByLoop(DateTime birthday)
{
var age = -1;
for (var date = birthday; date < DateTime.Today; date = date.AddYears(1))
{
age++;
}
return age;
}
Проверь это:
TimeSpan ts = DateTime.Now.Subtract(Birthdate);
age = (byte)(ts.TotalDays / 365.25);
Вот функция, которая мне хорошо служит. Никаких расчетов, очень просто.
public static string ToAge(this DateTime dob, DateTime? toDate = null)
{
if (!toDate.HasValue)
toDate = DateTime.Now;
var now = toDate.Value;
if (now.CompareTo(dob) < 0)
return "Future date";
int years = now.Year - dob.Year;
int months = now.Month - dob.Month;
int days = now.Day - dob.Day;
if (days < 0)
{
months--;
days = DateTime.DaysInMonth(dob.Year, dob.Month) - dob.Day + now.Day;
}
if (months < 0)
{
years--;
months = 12 + months;
}
return string.Format("{0} year(s), {1} month(s), {2} days(s)",
years,
months,
days);
}
А вот модульный тест:
[Test]
public void ToAgeTests()
{
var date = new DateTime(2000, 1, 1);
Assert.AreEqual("0 year(s), 0 month(s), 1 days(s)", new DateTime(1999, 12, 31).ToAge(date));
Assert.AreEqual("0 year(s), 0 month(s), 0 days(s)", new DateTime(2000, 1, 1).ToAge(date));
Assert.AreEqual("1 year(s), 0 month(s), 0 days(s)", new DateTime(1999, 1, 1).ToAge(date));
Assert.AreEqual("0 year(s), 11 month(s), 0 days(s)", new DateTime(1999, 2, 1).ToAge(date));
Assert.AreEqual("0 year(s), 10 month(s), 25 days(s)", new DateTime(1999, 2, 4).ToAge(date));
Assert.AreEqual("0 year(s), 10 month(s), 1 days(s)", new DateTime(1999, 2, 28).ToAge(date));
date = new DateTime(2000, 2, 15);
Assert.AreEqual("0 year(s), 0 month(s), 28 days(s)", new DateTime(2000, 1, 18).ToAge(date));
}
Это может быть так просто:
int age = DateTime.Now.AddTicks(0 - dob.Ticks).Year - 1;
Я бы просто сделал это:
DateTime birthDay = new DateTime(1990, 05, 23);
DateTime age = DateTime.Now - birthDay;
Таким образом вы можете рассчитать точный возраст человека с точностью до миллисекунды, если хотите.
Это не правильно. В вашем коде возраст будет TimeSpan. Не DateTime.
Проблема в том, что такой возраст, как «17 лет», не переводится напрямую в TimeSpan, потому что вы не знаете, какие из этих 17 лет были високосными.
Я использовал следующее для этой проблемы. Я знаю, что это не очень элегантно, но работает.
DateTime zeroTime = new DateTime(1, 1, 1);
var date1 = new DateTime(1983, 03, 04);
var date2 = DateTime.Now;
var dif = date2 - date1;
int years = (zeroTime + dif).Year - 1;
Log.DebugFormat("Years -->{0}", years);
public string GetAge(this DateTime birthdate, string ageStrinFormat = null)
{
var date = DateTime.Now.AddMonths(-birthdate.Month).AddDays(-birthdate.Day);
return string.Format(ageStrinFormat ?? "{0}/{1}/{2}",
(date.Year - birthdate.Year), date.Month, date.Day);
}
Это самый простой способ ответить на этот вопрос одной строкой.
DateTime Dob = DateTime.Parse("1985-04-24");
int Age = DateTime.MinValue.AddDays(DateTime.Now.Subtract(Dob).TotalHours/24 - 1).Year - 1;
Это также работает для високосных лет.
ваш ответ неверен на один день, он даст день рождения на день раньше
=== Распространенное высказывание (от месяцев до лет) ===
Если вы просто для общего пользования, вот код в качестве информации:
DateTime today = DateTime.Today;
DateTime bday = DateTime.Parse("2016-2-14");
int age = today.Year - bday.Year;
var unit = "";
if (bday > today.AddYears(-age))
{
age--;
}
if (age == 0) // Under one year old
{
age = today.Month - bday.Month;
age = age <= 0 ? (12 + age) : age; // The next year before birthday
age = today.Day - bday.Day >= 0 ? age : --age; // Before the birthday.day
unit = "month";
}
else {
unit = "year";
}
if (age > 1)
{
unit = unit + "s";
}
Результат теста, как показано ниже:
The birthday: 2016-2-14
2016-2-15 => age=0, unit=month;
2016-5-13 => age=2, unit=months;
2016-5-14 => age=3, unit=months;
2016-6-13 => age=3, unit=months;
2016-6-15 => age=4, unit=months;
2017-1-13 => age=10, unit=months;
2017-1-14 => age=11, unit=months;
2017-2-13 => age=11, unit=months;
2017-2-14 => age=1, unit=year;
2017-2-15 => age=1, unit=year;
2017-3-13 => age=1, unit=year;
2018-1-13 => age=1, unit=year;
2018-1-14 => age=1, unit=year;
2018-2-13 => age=1, unit=year;
2018-2-14 => age=2, unit=years;
Версия SQL:
declare @dd smalldatetime = '1980-04-01'
declare @age int = YEAR(GETDATE())-YEAR(@dd)
if (@dd> DATEADD(YYYY, -@age, GETDATE())) set @age = @age -1
print @age
Вау, мне пришлось здесь дать свой ответ ... На такой простой вопрос столько ответов.
private int CalcularIdade(DateTime dtNascimento)
{
var nHoje = Convert.ToInt32(DateTime.Today.ToString("yyyyMMdd"));
var nAniversario = Convert.ToInt32(dtNascimento.ToString("yyyyMMdd"));
double diff = (nHoje - nAniversario) / 10000;
var ret = Convert.ToInt32(Math.Truncate(diff));
return ret;
}
private int GetYearDiff(DateTime start, DateTime end)
{
int diff = end.Year - start.Year;
if (end.DayOfYear < start.DayOfYear) { diff -= 1; }
return diff;
}
[Fact]
public void GetYearDiff_WhenCalls_ShouldReturnCorrectYearDiff()
{
//arrange
var now = DateTime.Now;
//act
//assert
Assert.Equal(24, GetYearDiff(new DateTime(1992, 7, 9), now)); // passed
Assert.Equal(24, GetYearDiff(new DateTime(1992, now.Month, now.Day), now)); // passed
Assert.Equal(23, GetYearDiff(new DateTime(1992, 12, 9), now)); // passed
}
Я часто считаю по пальцам. Мне нужно посмотреть в календарь, чтобы понять, когда что-то изменится. Вот что я бы сделал в своем коде:
int AgeNow(DateTime birthday)
{
return AgeAt(DateTime.Now, birthday);
}
int AgeAt(DateTime now, DateTime birthday)
{
return AgeAt(now, birthday, CultureInfo.CurrentCulture.Calendar);
}
int AgeAt(DateTime now, DateTime birthday, Calendar calendar)
{
// My age has increased on the morning of my
// birthday even though I was born in the evening.
now = now.Date;
birthday = birthday.Date;
var age = 0;
if (now <= birthday) return age; // I am zero now if I am to be born tomorrow.
while (calendar.AddYears(birthday, age + 1) <= now)
{
age++;
}
return age;
}
Выполнение этого в LINQPad дает следующее:
PASSED: someone born on 28 February 1964 is age 4 on 28 February 1968
PASSED: someone born on 29 February 1964 is age 3 on 28 February 1968
PASSED: someone born on 31 December 2016 is age 0 on 01 January 2017
Код в LINQPad - здесь.
Простой код
var birthYear=1993;
var age = DateTime.Now.AddYears(-birthYear).Year;
Чтобы посчитать, сколько лет человеку,
DateTime dateOfBirth;
int ageInYears = DateTime.Now.Year - dateOfBirth.Year;
if (dateOfBirth > today.AddYears(-ageInYears )) ageInYears --;
вы только что скопировали первый ответ, что дает Патель?
Просто используйте:
(DateTime.Now - myDate).TotalHours / 8766.0
Текущая дата - myDate = TimeSpan, получите общее количество часов и разделите на общее количество часов в году и получите точный возраст / месяцы / дни ...
А как насчет високосных лет?
Вот самый простой способ рассчитать возраст человека. Подсчитать возраст человека довольно просто, и вот как это сделать! Чтобы код работал, вам нужен объект DateTime с именем BirthDate, содержащий дату рождения.
C#
// get the difference in years
int years = DateTime.Now.Year - BirthDate.Year;
// subtract another year if we're before the
// birth day in the current year
if (DateTime.Now.Month < BirthDate.Month ||
(DateTime.Now.Month == BirthDate.Month &&
DateTime.Now.Day < BirthDate.Day))
years--;
VB.NET
' get the difference in years
Dim years As Integer = DateTime.Now.Year - BirthDate.Year
' subtract another year if we're before the
' birth day in the current year
If DateTime.Now.Month < BirthDate.Month Or (DateTime.Now.Month = BirthDate.Month And DateTime.Now.Day < BirthDate.Day) Then
years = years - 1
End If
«Вот самый простой способ рассчитать чей-то возраст». Это действительно не самый простой способ. Используя Noda Time, это просто int years = Period.Between(birthDate, today).Years;.
@JonSkeet Честно говоря, Noda Time не является частью какого-либо стандарта .Net. Учитывая это, этот является один из самых простых способов рассчитать возраст человека в годах.
var birthDate = ... // DOB
var resultDate = DateTime.Now - birthDate;
Используя resultDate, вы можете применять свойства TimeSpan, как хотите.
Очень простой ответ
DateTime dob = new DateTime(1991, 3, 4);
DateTime now = DateTime.Now;
int dobDay = dob.Day, dobMonth = dob.Month;
int add = -1;
if (dobMonth < now.Month)
{
add = 0;
}
else if (dobMonth == now.Month)
{
if (dobDay <= now.Day)
{
add = 0;
}
else
{
add = -1;
}
}
else
{
add = -1;
}
int age = now.Year - dob.Year + add;
Должно быть описание вашего ответа для OP, а также для будущих читателей.
Ваш код не простой, это беспорядок. Я бы написал это, может быть, вы не верите, но код ниже делает то же самое (сейчас - dob). Год
int Age = new DateTime((DateTime.Now - BirthDate).Ticks).Year -1;
Console.WriteLine("Age {0}", Age);
Я думаю, что эту проблему можно решить более простым способом -
Класс может быть таким:
using System;
namespace TSA
{
class BirthDay
{
double ageDay;
public BirthDay(int day, int month, int year)
{
DateTime birthDate = new DateTime(year, month, day);
ageDay = (birthDate - DateTime.Now).TotalDays; //DateTime.UtcNow
}
internal int GetAgeYear()
{
return (int)Math.Truncate(ageDay / 365);
}
internal int GetAgeMonth()
{
return (int)Math.Truncate((ageDay % 365) / 30);
}
}
}
И звонки могут быть такими:
BirthDay b = new BirthDay(1,12,1990);
int year = b.GetAgeYear();
int month = b.GetAgeMonth();
Я ничего не знаю о DateTime, но могу сделать следующее:
using System;
public class Program
{
public static int getAge(int month, int day, int year) {
DateTime today = DateTime.Today;
int currentDay = today.Day;
int currentYear = today.Year;
int currentMonth = today.Month;
int age = 0;
if (currentMonth < month) {
age -= 1;
} else if (currentMonth == month) {
if (currentDay < day) {
age -= 1;
}
}
currentYear -= year;
age += currentYear;
return age;
}
public static void Main()
{
int ageInYears = getAge(8, 10, 2007);
Console.WriteLine(ageInYears);
}
}
Немного сбивает с толку, но если присмотреться к коду внимательнее, все будет иметь смысл.
что до сих пор упускали из виду все ответы, так это то, что это зависит от того, где человек родился и где он сейчас находится.