Ниже на самом деле наиболее точным способом вы можете сделать это, так как определение «1 месяц» меняется в зависимости от того, какой месяц она есть, и не другие ответы принять это во внимание! Если вы хотите получить дополнительную информацию о проблеме, которая не встроена в фреймворк, вы можете прочитать это сообщение: A Real Timespan Object With .Years & .Months (однако чтение этого сообщения не обязательно для понимания и использования функции ниже, оно работает на 100%, без присущих неточностей из приближения, которые другие любят использовать, и не стесняйтесь заменить функцию .ReverseIt встроенной функцией .Reverse, которую вы можете иметь в своей структуре (она здесь только для полноты).
Обратите внимание, что вы можете получить любое количество дат/времени, секунд & минут, секунд, минут и дней в любом месте до года (в котором будет 6 частей/сегментов). Если вы укажете два лучших, и вам будет больше года, он вернется «1 год и 3 месяца назад» и не вернет остальных, потому что вы запросили два сегмента. если ему всего несколько часов, тогда он вернется только «2 часа и 1 минута назад». Конечно, те же правила применяются, если вы укажете 1, 2, 3, 4, 5 или 6 segmets (максимум на 6, поскольку секунды, минуты, часы, дни, месяцы, годы составляют только 6 типов). Он также будет исправлять проблемы с грамматикой, такие как «минуты» и «минута», в зависимости от того, будет ли она 1 минута или более, одинаковой для всех типов, и генерируемая строка будет всегда грамматически корректной.
Вот некоторые примеры использования: bAllowSegments определяет, сколько сегментов, чтобы показать ... то есть: если 3, то вернуться строка будет (в качестве примера) ... "3 years, 2 months and 13 days"
(не будет включать в себя часы, минуты и секунды, когда вернутся 3 категории времени), если, однако, дата была более новой датой, например, что-то несколько дней назад, указав те же сегменты (3), вместо этого вернет "4 days, 1 hour and 13 minutes ago"
, поэтому это учитывает все!
если bAllowSegments 2 он вернется "3 years and 2 months"
и если 6 (максимальное значение) будет возвращать "3 years, 2 months, 13 days, 13 hours, 29 minutes and 9 seconds"
, но напомнить, что это будет NEVER RETURN
что-то вроде этого "0 years, 0 months, 0 days, 3 hours, 2 minutes and 13 seconds ago"
, как он понимает, что нет данных даты в 3 верхних сегментах и игнорируют их, даже если вы укажете 6 сегментов, так что не волнуйтесь :). Конечно, если в нем есть сегмент с 0, он учитывает это при формировании строки и будет отображаться как "3 days and 4 seconds ago"
и игнорирует часть «0 часов»! Наслаждайтесь и прокомментируйте, если хотите.
Public Function RealTimeUntilNow(ByVal dt As DateTime, Optional ByVal bAllowSegments As Byte = 2) As String
' bAllowSegments identifies how many segments to show... ie: if 3, then return string would be (as an example)...
' "3 years, 2 months and 13 days" the top 3 time categories are returned, if bAllowSegments is 2 it would return
' "3 years and 2 months" and if 6 (maximum value) would return "3 years, 2 months, 13 days, 13 hours, 29 minutes and 9 seconds"
Dim rYears, rMonths, rDays, rHours, rMinutes, rSeconds As Int16
Dim dtNow = DateTime.Now
Dim daysInBaseMonth = Date.DaysInMonth(dt.Year, dt.Month)
rYears = dtNow.Year - dt.Year
rMonths = dtNow.Month - dt.Month
If rMonths < 0 Then rMonths += 12 : rYears -= 1 ' add 1 year to months, and remove 1 year from years.
rDays = dtNow.Day - dt.Day
If rDays < 0 Then rDays += daysInBaseMonth : rMonths -= 1
rHours = dtNow.Hour - dt.Hour
If rHours < 0 Then rHours += 24 : rDays -= 1
rMinutes = dtNow.Minute - dt.Minute
If rMinutes < 0 Then rMinutes += 60 : rHours -= 1
rSeconds = dtNow.Second - dt.Second
If rSeconds < 0 Then rSeconds += 60 : rMinutes -= 1
' this is the display functionality
Dim sb As StringBuilder = New StringBuilder()
Dim iSegmentsAdded As Int16 = 0
If rYears > 0 Then sb.Append(rYears) : sb.Append(" year" & If(rYears <> 1, "s", "") & ", ") : iSegmentsAdded += 1
If bAllowSegments = iSegmentsAdded Then GoTo parseAndReturn
If rMonths > 0 Then sb.AppendFormat(rMonths) : sb.Append(" month" & If(rMonths <> 1, "s", "") & ", ") : iSegmentsAdded += 1
If bAllowSegments = iSegmentsAdded Then GoTo parseAndReturn
If rDays > 0 Then sb.Append(rDays) : sb.Append(" day" & If(rDays <> 1, "s", "") & ", ") : iSegmentsAdded += 1
If bAllowSegments = iSegmentsAdded Then GoTo parseAndReturn
If rHours > 0 Then sb.Append(rHours) : sb.Append(" hour" & If(rHours <> 1, "s", "") & ", ") : iSegmentsAdded += 1
If bAllowSegments = iSegmentsAdded Then GoTo parseAndReturn
If rMinutes > 0 Then sb.Append(rMinutes) : sb.Append(" minute" & If(rMinutes <> 1, "s", "") & ", ") : iSegmentsAdded += 1
If bAllowSegments = iSegmentsAdded Then GoTo parseAndReturn
If rSeconds > 0 Then sb.Append(rSeconds) : sb.Append(" second" & If(rSeconds <> 1, "s", "") & "") : iSegmentsAdded += 1
parseAndReturn:
' if the string is entirely empty, that means it was just posted so its less than a second ago, and an empty string getting passed will cause an error
' so we construct our own meaningful string which will still fit into the "Posted * ago " syntax...
If sb.ToString = "" Then sb.Append("less than 1 second")
Return ReplaceLast(sb.ToString.TrimEnd(" ", ",").ToString, ",", " and")
End Function
Конечно, вам нужна функция «ReplaceLast», которая принимает строку источника, и аргумент, указывающий, что необходимо заменить, а другой аргумент с указанием того, что вы хотите, чтобы заменить его, и это только заменяет последнее заполнение этой строки ... я включил мой, если у вас его нет или он не хочет его реализовывать, так что вот он, он будет работать «как есть» без каких-либо изменений. Я знаю, что функция reverseit больше не нужна (существует в .net), но функции ReplaceLast и ReverseIt переносятся с дней pre -.net, поэтому, пожалуйста, извините, как это датируется (все еще работает 100% tho, используется em более десяти лет, может гарантировать, что они являются ошибками бесплатно) ... :). веселит.
<Extension()> _
Public Function ReplaceLast(ByVal sReplacable As String, ByVal sReplaceWhat As String, ByVal sReplaceWith As String) As String
' let empty string arguments run, incase we dont know if we are sending and empty string or not.
sReplacable = sReplacable.ReverseIt
sReplacable = Replace(sReplacable, sReplaceWhat.ReverseIt, sReplaceWith.ReverseIt, , 1) ' only does first item on reversed version!
Return sReplacable.ReverseIt.ToString
End Function
<Extension()> _
Public Function ReverseIt(ByVal strS As String, Optional ByVal n As Integer = -1) As String
Dim strTempX As String = "", intI As Integer
If n > strS.Length Or n = -1 Then n = strS.Length
For intI = n To 1 Step -1
strTempX = strTempX + Mid(strS, intI, 1)
Next intI
ReverseIt = strTempX + Right(strS, Len(strS) - n)
End Function
Что вы ожидаете на 25 декабря 2009 - 6 окт 2009? –
Как вы определяете TimeSpan в месяцах? – Aliostad
@Aliostad - Без дат вы можете определить месяц как 30 дней и быть довольно точным. – ChaosPandion