Was ist, wenn Sie eine relative Zeit von jetzt bis in die Zukunft berechnen möchten?
Jhonny D. Cano -Leftware-
2
moment.js ist eine sehr schöne Datumsanalyse-Bibliothek. Sie können diese je nach Ihren Anforderungen (Server- oder Client-Seite) verwenden. Nur zu Ihrer Information, weil es hier niemand erwähnt hat
Ich hasse solche Konstanten aus Leidenschaft. Sieht das für jemanden falsch aus? Thread.Sleep(1 * MINUTE)? Weil es um den Faktor 1000 falsch ist.
Roman Starkov
31
const int SECOND = 1;So seltsam ist eine Sekunde eine Sekunde.
Seriousdev
62
Diese Art von Code ist kaum zu lokalisieren. Wenn Ihre App nur auf Englisch bleiben muss, ist das in Ordnung. Aber wenn Sie den Sprung in andere Sprachen schaffen, werden Sie sich dafür hassen, dass Sie Logik wie diese machen. Nur damit Sie wissen ...
Nik Reiman
73
Ich denke, wenn die Konstanten umbenannt würden, um den Wert, der in ihnen enthalten ist, genau zu beschreiben, wäre es einfacher zu verstehen. Also SecondsPerMinute = 60; MinutesPerHour = 60; SecondsPerHour = MinutesPerHour * SecondsPerHour; usw. Wenn Sie es nur MINUTE = 60 nennen, kann der Leser den Wert nicht bestimmen.
Slolife
14
Warum kümmert sich niemand (außer Joe) um den falschen Wert für "Gestern" oder "Vor Tagen" ??? Gestern ist keine Stundenberechnung, sondern eine tägliche Berechnung. Also ja, dies ist zumindest in zwei häufigen Fällen ein falscher Code.
Seb, Wenn Sie Javascript deaktiviert haben, wird die Zeichenfolge angezeigt, die Sie ursprünglich zwischen die abbr-Tags eingefügt haben. In der Regel ist dies nur ein Datum oder eine Uhrzeit in einem beliebigen Format. Timeago verschlechtert sich anmutig. Einfacher geht es nicht.
Heh, danke Rob. Das ist okay. Es ist kaum wahrnehmbar, insbesondere wenn sich während des Übergangs nur eine Nummer ändert, obwohl SO-Seiten viele Zeitstempel haben. Ich hätte gedacht, er hätte zumindest die Vorteile des Seiten-Cachings geschätzt, selbst wenn er sich dafür entschieden hätte, automatische Updates zu vermeiden. Ich bin sicher, Jeff hätte Feedback geben können, um auch das Plugin zu verbessern. Ich tröste mich mit dem Wissen, dass Websites wie arstechnica.com es verwenden.
Ryan McGeary
19
@ Rob Fonseca-Ensor - jetzt bringt es mich auch zum Weinen. Wie kann ein Update einmal pro Minute durchgeführt werden, um genaue Informationen anzuzeigen, die in irgendeiner Weise damit zusammenhängen, dass Text einmal pro Sekunde blinkt?
Daniel Earwicker
25
Die Frage bezieht sich auf C #. Ich sehe nicht, wie relevant ein jQuery-Plugin ist.
"<48 * 60 * 60s" ist eine eher unkonventionelle Definition für "gestern". Wenn es am Mittwoch 9 Uhr morgens ist, würden Sie 9:01 Uhr am Montag wirklich als "gestern" betrachten. Ich hätte gedacht, ein Algorithmus für gestern oder "vor n Tagen" sollte vor / nach Mitternacht in Betracht gezogen werden.
Joe
139
Compiler sind normalerweise ziemlich gut darin, konstante Ausdrücke wie 24 * 60 * 60 vorab zu berechnen. Sie können diese also direkt verwenden, anstatt sie selbst auf 86400 zu berechnen und den ursprünglichen Ausdruck in Kommentare zu setzen
zvolkov
11
@bzlm Ich glaube, ich habe es für ein Projekt getan, an dem ich gearbeitet habe. Meine Motivation hier war es, andere darauf aufmerksam zu machen, dass Wochen in diesem Codebeispiel weggelassen wurden. Wie das geht, kam mir ziemlich direkt vor.
Jray
9
Ich denke, dass ein guter Weg, um den Algorithmus zu verbessern, darin besteht, 2 Einheiten wie "vor 2 Monaten 21 Tagen" und "vor 1 Stunde 40 Minuten" anzuzeigen, um die Genauigkeit zu erhöhen.
Evgeny Levin
5
@ Jeffy, Sie haben die Berechnung für das Schaltjahr und die damit verbundenen Prüfungen verpasst
Saboor Awan
92
publicstaticstringRelativeDate(DateTime theDate){Dictionary<long,string> thresholds =newDictionary<long,string>();int minute =60;int hour =60* minute;int day =24* hour;
thresholds.Add(60,"{0} seconds ago");
thresholds.Add(minute *2,"a minute ago");
thresholds.Add(45* minute,"{0} minutes ago");
thresholds.Add(120* minute,"an hour ago");
thresholds.Add(day,"{0} hours ago");
thresholds.Add(day *2,"yesterday");
thresholds.Add(day *30,"{0} days ago");
thresholds.Add(day *365,"{0} months ago");
thresholds.Add(long.MaxValue,"{0} years ago");long since =(DateTime.Now.Ticks- theDate.Ticks)/10000000;foreach(long threshold in thresholds.Keys){if(since < threshold){TimeSpan t =newTimeSpan((DateTime.Now.Ticks- theDate.Ticks));returnstring.Format(thresholds[threshold],(t.Days>365? t.Days/365:(t.Days>0? t.Days:(t.Hours>0? t.Hours:(t.Minutes>0? t.Minutes:(t.Seconds>0? t.Seconds:0))))).ToString());}}return"";}
Ich bevorzuge diese Version wegen ihrer Prägnanz und der Fähigkeit, neue Tickpunkte hinzuzufügen. Dies könnte mit einer Latest()Erweiterung auf Timespan anstelle dieses langen 1-Liner gekapselt werden , aber der Kürze halber reicht dies aus.
Dies behebt das Problem vor einer Stunde, vor 1 Stunde, indem eine Stunde angegeben wird, bis 2 Stunden vergangen sind
Ich bekomme alle möglichen Probleme mit dieser Funktion, zum Beispiel wenn Sie 'theDate = DateTime.Now.AddMinutes (-40);' verspotten. Ich erhalte 'vor 40 Stunden', aber mit Michaels Refactormycode-Antwort wird es korrekt bei 'vor 40 Minuten' zurückgegeben?
GONeale
Ich denke, Sie vermissen eine Null, versuchen Sie: long Since = (DateTime.Now.Ticks - theDate.Ticks) / 10000000;
Robnardo
8
Hmm, obwohl dieser Code möglicherweise funktioniert, ist es falsch und ungültig anzunehmen, dass die Reihenfolge der Schlüssel im Wörterbuch in einer bestimmten Reihenfolge liegt. Das Wörterbuch verwendet den Object.GetHashCode (), der kein langes, sondern ein int! Zurückgibt. Wenn Sie möchten, dass diese sortiert werden, sollten Sie eine SortedList <long, string> verwenden. Was ist falsch daran, dass die Schwellenwerte in einer Reihe von if / else if /.../ else ausgewertet werden? Sie erhalten die gleiche Anzahl von Vergleichen. Zu Ihrer Information, der Hash für lange. MaxValue ist der gleiche wie int.MinValue!
CodeMonkeyKing
OP vergessen t.Days> 30? t.Days / 30:
Lars Holm Jensen
Um das von @CodeMonkeyKing erwähnte Problem zu beheben, können Sie ein SortedDictionarystatt eines einfachen verwenden Dictionary: Die Verwendung ist dieselbe, stellt jedoch sicher, dass die Schlüssel sortiert sind. Aber selbst dann hat der Algorithmus Fehler, da RelativeDate(DateTime.Now.AddMonths(-3).AddDays(-3))kehrt „95 Monate vor“ , unabhängig , welche Wörterbuchtyp Sie verwenden, was falsch ist (es sollte zurückkehren „3 Monate zuvor“ oder „4 Monate“ je nachdem , welcher Schwelle Sie‘ Wiederverwendung) - auch wenn -3 im letzten Jahr kein Datum erstellt (ich habe dies im Dezember getestet, daher sollte es in diesem Fall nicht passieren).
staticreadonlySortedList<double,Func<TimeSpan,string>> offsets =newSortedList<double,Func<TimeSpan,string>>{{0.75, _ =>"less than a minute"},{1.5, _ =>"about a minute"},{45, x => $"{x.TotalMinutes:F0} minutes"},{90, x =>"about an hour"},{1440, x => $"about {x.TotalHours:F0} hours"},{2880, x =>"a day"},{43200, x => $"{x.TotalDays:F0} days"},{86400, x =>"about a month"},{525600, x => $"{x.TotalDays / 30:F0} months"},{1051200, x =>"about a year"},{double.MaxValue, x => $"{x.TotalDays / 365:F0} years"}};publicstaticstringToRelativeDate(thisDateTime input){TimeSpan x =DateTime.Now- input;stringSuffix= x.TotalMinutes>0?" ago":" from now";
x =newTimeSpan(Math.Abs(x.Ticks));return offsets.First(n => x.TotalMinutes< n.Key).Value(x)+Suffix;}
das ist sehr schön IMO :) Dies könnte auch als Erweiterungsmethode überarbeitet werden? Könnte das Wörterbuch statisch werden, sodass es nur einmal erstellt und von da an referenziert wird?
Sie möchten dieses Wörterbuch wahrscheinlich in ein Feld ziehen, um die Instanziierung und die GC-Abwanderung zu reduzieren. Sie müssten sich ändern Func<string>zu Func<double>.
Drew Noakes
49
Hier ist eine Implementierung, die ich als Erweiterungsmethode zur DateTime-Klasse hinzugefügt habe. Sie behandelt sowohl zukünftige als auch vergangene Daten und bietet eine Annäherungsoption, mit der Sie den gewünschten Detaillierungsgrad angeben können ("vor 3 Stunden" gegenüber "3 Stunden". Vor 23 Minuten, 12 Sekunden "):
usingSystem.Text;/// <summary>/// Compares a supplied date to the current date and generates a friendly English /// comparison ("5 days ago", "5 days from now")/// </summary>/// <param name="date">The date to convert</param>/// <param name="approximate">When off, calculate timespan down to the second./// When on, approximate to the largest round unit of time.</param>/// <returns></returns>publicstaticstringToRelativeDateString(thisDateTimevalue,bool approximate){StringBuilder sb =newStringBuilder();string suffix =(value>DateTime.Now)?" from now":" ago";TimeSpan timeSpan =newTimeSpan(Math.Abs(DateTime.Now.Subtract(value).Ticks));if(timeSpan.Days>0){
sb.AppendFormat("{0} {1}", timeSpan.Days,(timeSpan.Days>1)?"days":"day");if(approximate)return sb.ToString()+ suffix;}if(timeSpan.Hours>0){
sb.AppendFormat("{0}{1} {2}",(sb.Length>0)?", ":string.Empty,
timeSpan.Hours,(timeSpan.Hours>1)?"hours":"hour");if(approximate)return sb.ToString()+ suffix;}if(timeSpan.Minutes>0){
sb.AppendFormat("{0}{1} {2}",(sb.Length>0)?", ":string.Empty,
timeSpan.Minutes,(timeSpan.Minutes>1)?"minutes":"minute");if(approximate)return sb.ToString()+ suffix;}if(timeSpan.Seconds>0){
sb.AppendFormat("{0}{1} {2}",(sb.Length>0)?", ":string.Empty,
timeSpan.Seconds,(timeSpan.Seconds>1)?"seconds":"second");if(approximate)return sb.ToString()+ suffix;}if(sb.Length==0)return"right now";
sb.Append(suffix);return sb.ToString();}
freundlicher Hinweis: Installieren Sie auf .net 4.5 oder höher nicht den vollständigen Humanizer ... installieren Sie nur Humanizer.Core Teil davon .. weil andere Sprachpakete in dieser Version nicht unterstützt werden
Ahmad
So nützlich! Diese Antwort muss in dieser Liste viel höher sein. Wenn ich 100 Stimmen hätte, würde ich es geben. Anscheinend (aus JS-Land) war die Suche nach diesem Paket nicht einfach.
Kumarshar
29
@ Jeff
IMHO deine scheint ein wenig lang. Mit Unterstützung für "gestern" und "Jahre" scheint es jedoch etwas robuster zu sein. Nach meiner Erfahrung ist es jedoch am wahrscheinlichsten, dass die Person den Inhalt in den ersten 30 Tagen anzeigt, wenn dies verwendet wird. Es sind nur die wirklich Hardcore-Leute, die danach kommen. Deshalb entscheide ich mich normalerweise dafür, dies kurz und einfach zu halten.
Dies ist die Methode, die ich derzeit auf einer meiner Websites verwende. Dies gibt nur einen relativen Tag, eine relative Stunde und eine relative Uhrzeit zurück. Und dann muss der Benutzer in der Ausgabe auf "vor" klatschen.
Ein paar Jahre zu spät zur Party, aber ich musste dies sowohl für vergangene als auch für zukünftige Daten tun, also kombinierte ich Jeffs und Vincents dazu. Es ist eine ternarytastische Extravaganz! :) :)
publicstaticclassDateTimeHelper{privateconstint SECOND =1;privateconstint MINUTE =60* SECOND;privateconstint HOUR =60* MINUTE;privateconstint DAY =24* HOUR;privateconstint MONTH =30* DAY;/// <summary>/// Returns a friendly version of the provided DateTime, relative to now. E.g.: "2 days ago", or "in 6 months"./// </summary>/// <param name="dateTime">The DateTime to compare to Now</param>/// <returns>A friendly string</returns>publicstaticstringGetFriendlyRelativeTime(DateTime dateTime){if(DateTime.UtcNow.Ticks== dateTime.Ticks){return"Right now!";}bool isFuture =(DateTime.UtcNow.Ticks< dateTime.Ticks);var ts =DateTime.UtcNow.Ticks< dateTime.Ticks?newTimeSpan(dateTime.Ticks-DateTime.UtcNow.Ticks):newTimeSpan(DateTime.UtcNow.Ticks- dateTime.Ticks);double delta = ts.TotalSeconds;if(delta <1* MINUTE){return isFuture ?"in "+(ts.Seconds==1?"one second": ts.Seconds+" seconds"): ts.Seconds==1?"one second ago": ts.Seconds+" seconds ago";}if(delta <2* MINUTE){return isFuture ?"in a minute":"a minute ago";}if(delta <45* MINUTE){return isFuture ?"in "+ ts.Minutes+" minutes": ts.Minutes+" minutes ago";}if(delta <90* MINUTE){return isFuture ?"in an hour":"an hour ago";}if(delta <24* HOUR){return isFuture ?"in "+ ts.Hours+" hours": ts.Hours+" hours ago";}if(delta <48* HOUR){return isFuture ?"tomorrow":"yesterday";}if(delta <30* DAY){return isFuture ?"in "+ ts.Days+" days": ts.Days+" days ago";}if(delta <12* MONTH){int months =Convert.ToInt32(Math.Floor((double)ts.Days/30));return isFuture ?"in "+(months <=1?"one month": months +" months"): months <=1?"one month ago": months +" months ago";}else{int years =Convert.ToInt32(Math.Floor((double)ts.Days/365));return isFuture ?"in "+(years <=1?"one year": years +" years"): years <=1?"one year ago": years +" years ago";}}}
Angesichts der Tatsache, dass die Welt und ihr Ehemann anscheinend Codebeispiele veröffentlichen, habe ich vor einiger Zeit Folgendes geschrieben, basierend auf einigen dieser Antworten.
Ich hatte ein spezielles Bedürfnis, dass dieser Code lokalisierbar ist. Ich habe also zwei Klassen - Grammardie die lokalisierbaren Begriffe angeben und FuzzyDateExtensionseine Reihe von Erweiterungsmethoden enthalten. Ich musste mich nicht mit zukünftigen Datenzeiten befassen, daher wird kein Versuch unternommen, sie mit diesem Code zu behandeln.
Ich habe einen Teil des XMLdocs in der Quelle belassen, aber der Kürze halber die meisten (wo sie offensichtlich wären) entfernt. Ich habe auch nicht jedes Klassenmitglied hier aufgenommen:
publicclassGrammar{/// <summary> Gets or sets the term for "just now". </summary>publicstringJustNow{get;set;}/// <summary> Gets or sets the term for "X minutes ago". </summary>/// <remarks>/// This is a <see cref="String.Format"/> pattern, where <c>{0}</c>/// is the number of minutes./// </remarks>publicstringMinutesAgo{get;set;}publicstringOneHourAgo{get;set;}publicstringHoursAgo{get;set;}publicstringYesterday{get;set;}publicstringDaysAgo{get;set;}publicstringLastMonth{get;set;}publicstringMonthsAgo{get;set;}publicstringLastYear{get;set;}publicstringYearsAgo{get;set;}/// <summary> Gets or sets the term for "ages ago". </summary>publicstringAgesAgo{get;set;}/// <summary>/// Gets or sets the threshold beyond which the fuzzy date should be/// considered "ages ago"./// </summary>publicTimeSpanAgesAgoThreshold{get;set;}/// <summary>/// Initialises a new <see cref="Grammar"/> instance with the/// specified properties./// </summary>privatevoidInitialise(string justNow,string minutesAgo,string oneHourAgo,string hoursAgo,string yesterday,string daysAgo,string lastMonth,string monthsAgo,string lastYear,string yearsAgo,string agesAgo,TimeSpan agesAgoThreshold){...}}
Eines der wichtigsten Dinge , die ich als auch Lokalisierung, war , dass „heute“ würde nur bedeuten , „dieses Kalendertag“, so das erreichen wollte IsToday, IsThisMonth, IsThisYearschauen Methoden wie folgt aus :
Ich dachte, ich würde es mit Klassen und Polymorphismus versuchen. Ich hatte eine frühere Iteration, bei der Unterklassen verwendet wurden, die viel zu viel Aufwand verursachten. Ich habe zu einem flexibleren Objektmodell für Delegierte / öffentliche Objekte gewechselt, das erheblich besser ist. Mein Code ist etwas genauer, ich wünschte, ich könnte einen besseren Weg finden, um "vor Monaten" zu generieren, der nicht zu überentwickelt schien.
Ich denke, ich würde mich immer noch an Jeffs Wenn-Dann-Kaskade halten, weil es weniger Code und einfacher ist (es ist definitiv einfacher sicherzustellen, dass es wie erwartet funktioniert).
Für den folgenden Code gibt PrintRelativeTime.GetRelativeTimeMessage (vor TimeSpan) die relative Zeitnachricht zurück (z. B. "gestern").
publicclassRelativeTimeRange:IComparable{publicTimeSpanUpperBound{get;set;}publicdelegatestringRelativeTimeTextDelegate(TimeSpan timeDelta);publicRelativeTimeTextDelegateMessageCreator{get;set;}publicintCompareTo(object obj){if(!(obj isRelativeTimeRange)){return1;}// note that this sorts in reverse order to the way you'd expect, // this saves having to reverse a list laterreturn(obj asRelativeTimeRange).UpperBound.CompareTo(UpperBound);}}publicclassPrintRelativeTime{privatestaticList<RelativeTimeRange> timeRanges;staticPrintRelativeTime(){
timeRanges =newList<RelativeTimeRange>{newRelativeTimeRange{UpperBound=TimeSpan.FromSeconds(1),MessageCreator=(delta)=>{return"one second ago";}},newRelativeTimeRange{UpperBound=TimeSpan.FromSeconds(60),MessageCreator=(delta)=>{return delta.Seconds+" seconds ago";}},newRelativeTimeRange{UpperBound=TimeSpan.FromMinutes(2),MessageCreator=(delta)=>{return"one minute ago";}},newRelativeTimeRange{UpperBound=TimeSpan.FromMinutes(60),MessageCreator=(delta)=>{return delta.Minutes+" minutes ago";}},newRelativeTimeRange{UpperBound=TimeSpan.FromHours(2),MessageCreator=(delta)=>{return"one hour ago";}},newRelativeTimeRange{UpperBound=TimeSpan.FromHours(24),MessageCreator=(delta)=>{return delta.Hours+" hours ago";}},newRelativeTimeRange{UpperBound=TimeSpan.FromDays(2),MessageCreator=(delta)=>{return"yesterday";}},newRelativeTimeRange{UpperBound=DateTime.Now.Subtract(DateTime.Now.AddMonths(-1)),MessageCreator=(delta)=>{return delta.Days+" days ago";}},newRelativeTimeRange{UpperBound=DateTime.Now.Subtract(DateTime.Now.AddMonths(-2)),MessageCreator=(delta)=>{return"one month ago";}},newRelativeTimeRange{UpperBound=DateTime.Now.Subtract(DateTime.Now.AddYears(-1)),MessageCreator=(delta)=>{return(int)Math.Floor(delta.TotalDays/30)+" months ago";}},newRelativeTimeRange{UpperBound=DateTime.Now.Subtract(DateTime.Now.AddYears(-2)),MessageCreator=(delta)=>{return"one year ago";}},newRelativeTimeRange{UpperBound=TimeSpan.MaxValue,MessageCreator=(delta)=>{return(int)Math.Floor(delta.TotalDays/365.24D)+" years ago";}}};
timeRanges.Sort();}publicstaticstringGetRelativeTimeMessage(TimeSpan ago){RelativeTimeRange postRelativeDateRange = timeRanges[0];foreach(var timeRange in timeRanges){if(ago.CompareTo(timeRange.UpperBound)<=0){
postRelativeDateRange = timeRange;}}return postRelativeDateRange.MessageCreator(ago);}}
StriplingWarrior: Einfaches Lesen und Ändern im Vergleich zu einer switch-Anweisung oder einem Stapel von if / else-Anweisungen. Das statische Wörterbuch bedeutet, dass es und die Func <,> -Objekte nicht jedes Mal erstellt werden müssen, wenn ToRelativeDate verwendet werden soll. Es wird nur einmal erstellt, verglichen mit dem, den ich in meiner Antwort verlinkt habe.
Chris Charabaruk
Aha. Ich habe nur gedacht, da in der Dokumentation zu Dictionary"Die Reihenfolge, in der die Artikel zurückgegeben werden, ist nicht definiert" ( msdn.microsoft.com/en-us/library/xfhwa508.aspx ) angegeben ist, dass dies möglicherweise nicht die beste zu verwendende Datenstruktur ist wenn Sie sich weniger für Suchzeiten interessieren als dafür, dass die Dinge in Ordnung bleiben.
StriplingWarrior
StriplingWarrior: Ich glaube, LINQ berücksichtigt dies, wenn es mit Dictionarys verwendet wird. Wenn Sie sich immer noch unwohl fühlen, können Sie es verwenden SortedDictionary, aber meine eigene Erfahrung zeigt, dass dies unnötig ist.
Chris Charabaruk
12
Wenn Sie die Zeitzone des Betrachters kennen, ist es möglicherweise klarer, Kalendertage auf der Tagesskala zu verwenden. Ich bin mit den .NET-Bibliotheken nicht vertraut, daher weiß ich leider nicht, wie Sie das in C # tun würden.
Auf Verbraucherseiten können Sie auch unter einer Minute von Hand wavier sein. "Vor weniger als einer Minute" oder "gerade jetzt" könnte gut genug sein.
Die Frage ist mit C # gekennzeichnet . Warum dieser Java-Code ? IMHO, gilt nur C # -Code
Kiquenet
9
@ Jeff
var ts =newTimeSpan(DateTime.UtcNow.Ticks- dt.Ticks);
Doing eine Subtraktion auf DateTimekehrt ein TimeSpanohnehin.
Also kannst du es einfach tun
(DateTime.UtcNow- dt).TotalSeconds
Ich bin auch überrascht zu sehen, wie die Konstanten von Hand multipliziert und dann Kommentare mit den Multiplikationen hinzugefügt werden. War das eine fehlgeleitete Optimierung?
Hier ist der Algorithmus, den der Stackoverflow verwendet, der jedoch im perlischen Pseudocode mit einer Fehlerbehebung (keine "vor einer Stunde") präziser umgeschrieben wurde. Die Funktion benötigt eine (positive) Anzahl von Sekunden und gibt eine menschenfreundliche Zeichenfolge wie "vor 3 Stunden" oder "gestern" zurück.
publicstaticstringTimeAgo(thisDateTime dateTime){string result =string.Empty;var timeSpan =DateTime.Now.Subtract(dateTime);if(timeSpan <=TimeSpan.FromSeconds(60)){
result =string.Format("{0} seconds ago", timeSpan.Seconds);}elseif(timeSpan <=TimeSpan.FromMinutes(60)){
result = timeSpan.Minutes>1?String.Format("about {0} minutes ago", timeSpan.Minutes):"about a minute ago";}elseif(timeSpan <=TimeSpan.FromHours(24)){
result = timeSpan.Hours>1?String.Format("about {0} hours ago", timeSpan.Hours):"about an hour ago";}elseif(timeSpan <=TimeSpan.FromDays(30)){
result = timeSpan.Days>1?String.Format("about {0} days ago", timeSpan.Days):"yesterday";}elseif(timeSpan <=TimeSpan.FromDays(365)){
result = timeSpan.Days>30?String.Format("about {0} months ago", timeSpan.Days/30):"about a month ago";}else{
result = timeSpan.Days>365?String.Format("about {0} years ago", timeSpan.Days/365):"about a year ago";}return result;}
Oder verwenden Sie das jQuery-Plugin mit der Razor-Erweiterung von Timeago.
Sie können die serverseitige Last reduzieren, indem Sie diese Logik clientseitig ausführen. Zeigen Sie die Quelle auf einigen Digg-Seiten als Referenz an. Der Server gibt einen Epochenzeitwert aus, der von Javascript verarbeitet wird. Auf diese Weise müssen Sie die Zeitzone des Endbenutzers nicht verwalten. Der neue serverseitige Code wäre ungefähr so:
Das habe ich von einem Blog von Bill Gates bekommen. Ich muss es in meinem Browserverlauf finden und gebe Ihnen den Link.
Der Javascript-Code, um dasselbe zu tun (wie angefordert):
function posted(t){var now =newDate();var diff = parseInt((now.getTime()-Date.parse(t))/1000);if(diff <60){return'less than a minute ago';}elseif(diff <120){return'about a minute ago';}elseif(diff <(2700)){return(parseInt(diff /60)).toString()+' minutes ago';}elseif(diff <(5400)){return'about an hour ago';}elseif(diff <(86400)){return'about '+(parseInt(diff /3600)).toString()+' hours ago';}elseif(diff <(172800)){return'1 day ago';}else{return(parseInt(diff /86400)).toString()+' days ago';}}
Ich denke, es gibt bereits eine Reihe von Antworten zu diesem Beitrag, aber man kann diese verwenden, die einfach wie das Plugin zu verwenden und auch für Programmierer leicht lesbar ist. Senden Sie Ihr bestimmtes Datum und erhalten Sie den Wert in Zeichenfolgenform:
Antworten:
Jeff, dein Code ist nett, könnte aber mit Konstanten klarer sein (wie in Code Complete vorgeschlagen).
quelle
Thread.Sleep(1 * MINUTE)
? Weil es um den Faktor 1000 falsch ist.const int SECOND = 1;
So seltsam ist eine Sekunde eine Sekunde.jquery.timeago Plugin
Jeff, da Stack Overflow jQuery häufig verwendet, empfehle ich das Plugin jquery.timeago .
Leistungen:
Hängen Sie es einfach an Ihre Zeitstempel auf DOM bereit:
Dadurch werden alle
abbr
Elemente mit einer Zeitklasse und einem ISO 8601- Zeitstempel im Titel gedreht:in so etwas:
was ergibt: vor 4 Monaten. Mit der Zeit werden die Zeitstempel automatisch aktualisiert.
Haftungsausschluss: Ich habe dieses Plugin geschrieben, bin also voreingenommen.
quelle
So mache ich es
Vorschläge? Bemerkungen? Möglichkeiten, diesen Algorithmus zu verbessern?
quelle
Ich bevorzuge diese Version wegen ihrer Prägnanz und der Fähigkeit, neue Tickpunkte hinzuzufügen. Dies könnte mit einer
Latest()
Erweiterung auf Timespan anstelle dieses langen 1-Liner gekapselt werden , aber der Kürze halber reicht dies aus. Dies behebt das Problem vor einer Stunde, vor 1 Stunde, indem eine Stunde angegeben wird, bis 2 Stunden vergangen sindquelle
SortedDictionary
statt eines einfachen verwendenDictionary
: Die Verwendung ist dieselbe, stellt jedoch sicher, dass die Schlüssel sortiert sind. Aber selbst dann hat der Algorithmus Fehler, daRelativeDate(DateTime.Now.AddMonths(-3).AddDays(-3))
kehrt „95 Monate vor“ , unabhängig , welche Wörterbuchtyp Sie verwenden, was falsch ist (es sollte zurückkehren „3 Monate zuvor“ oder „4 Monate“ je nachdem , welcher Schwelle Sie‘ Wiederverwendung) - auch wenn -3 im letzten Jahr kein Datum erstellt (ich habe dies im Dezember getestet, daher sollte es in diesem Fall nicht passieren).Hier eine Neufassung von Jeffs Script für PHP:
quelle
http://refactormycode.com/codes/493-twitter-esque-relative-dates
C # 6 Version:
quelle
Func<string>
zuFunc<double>
.Hier ist eine Implementierung, die ich als Erweiterungsmethode zur DateTime-Klasse hinzugefügt habe. Sie behandelt sowohl zukünftige als auch vergangene Daten und bietet eine Annäherungsoption, mit der Sie den gewünschten Detaillierungsgrad angeben können ("vor 3 Stunden" gegenüber "3 Stunden". Vor 23 Minuten, 12 Sekunden "):
quelle
Ich würde empfehlen, dies auch auf der Client-Seite zu berechnen. Weniger Arbeit für den Server.
Das Folgende ist die Version, die ich benutze (von Zach Leatherman)
quelle
Es gibt auch ein Paket namens Humanizr auf Nuget, das wirklich gut funktioniert und in der .NET Foundation enthalten ist.
Scott Hanselman hat einen Artikel darüber in seinem Blog
quelle
@ Jeff
IMHO deine scheint ein wenig lang. Mit Unterstützung für "gestern" und "Jahre" scheint es jedoch etwas robuster zu sein. Nach meiner Erfahrung ist es jedoch am wahrscheinlichsten, dass die Person den Inhalt in den ersten 30 Tagen anzeigt, wenn dies verwendet wird. Es sind nur die wirklich Hardcore-Leute, die danach kommen. Deshalb entscheide ich mich normalerweise dafür, dies kurz und einfach zu halten.
Dies ist die Methode, die ich derzeit auf einer meiner Websites verwende. Dies gibt nur einen relativen Tag, eine relative Stunde und eine relative Uhrzeit zurück. Und dann muss der Benutzer in der Ausgabe auf "vor" klatschen.
quelle
Ein paar Jahre zu spät zur Party, aber ich musste dies sowohl für vergangene als auch für zukünftige Daten tun, also kombinierte ich Jeffs und Vincents dazu. Es ist eine ternarytastische Extravaganz! :) :)
quelle
Gibt es eine einfache Möglichkeit, dies in Java zu tun? Die
java.util.Date
Klasse scheint eher begrenzt.Hier ist meine schnelle und schmutzige Java-Lösung:
quelle
iPhone Objective-C-Version
quelle
Angesichts der Tatsache, dass die Welt und ihr Ehemann anscheinend Codebeispiele veröffentlichen, habe ich vor einiger Zeit Folgendes geschrieben, basierend auf einigen dieser Antworten.
Ich hatte ein spezielles Bedürfnis, dass dieser Code lokalisierbar ist. Ich habe also zwei Klassen -
Grammar
die die lokalisierbaren Begriffe angeben undFuzzyDateExtensions
eine Reihe von Erweiterungsmethoden enthalten. Ich musste mich nicht mit zukünftigen Datenzeiten befassen, daher wird kein Versuch unternommen, sie mit diesem Code zu behandeln.Ich habe einen Teil des XMLdocs in der Quelle belassen, aber der Kürze halber die meisten (wo sie offensichtlich wären) entfernt. Ich habe auch nicht jedes Klassenmitglied hier aufgenommen:
Die
FuzzyDateString
Klasse enthält:Eines der wichtigsten Dinge , die ich als auch Lokalisierung, war , dass „heute“ würde nur bedeuten , „dieses Kalendertag“, so das erreichen wollte
IsToday
,IsThisMonth
,IsThisYear
schauen Methoden wie folgt aus :und die Rundungsmethoden sind wie folgt (ich habe aufgenommen
RoundedMonths
, da das etwas anders ist):Ich hoffe, die Leute finden das nützlich und / oder interessant: o)
quelle
In PHP mache ich das so:
quelle
mit Fluent DateTime
quelle
Ich dachte, ich würde es mit Klassen und Polymorphismus versuchen. Ich hatte eine frühere Iteration, bei der Unterklassen verwendet wurden, die viel zu viel Aufwand verursachten. Ich habe zu einem flexibleren Objektmodell für Delegierte / öffentliche Objekte gewechselt, das erheblich besser ist. Mein Code ist etwas genauer, ich wünschte, ich könnte einen besseren Weg finden, um "vor Monaten" zu generieren, der nicht zu überentwickelt schien.
Ich denke, ich würde mich immer noch an Jeffs Wenn-Dann-Kaskade halten, weil es weniger Code und einfacher ist (es ist definitiv einfacher sicherzustellen, dass es wie erwartet funktioniert).
Für den folgenden Code gibt PrintRelativeTime.GetRelativeTimeMessage (vor TimeSpan) die relative Zeitnachricht zurück (z. B. "gestern").
quelle
Das gleiche wie eine andere Antwort auf diese Frage, jedoch als Erweiterungsmethode mit einem statischen Wörterbuch.
quelle
Dictionary
"Die Reihenfolge, in der die Artikel zurückgegeben werden, ist nicht definiert" ( msdn.microsoft.com/en-us/library/xfhwa508.aspx ) angegeben ist, dass dies möglicherweise nicht die beste zu verwendende Datenstruktur ist wenn Sie sich weniger für Suchzeiten interessieren als dafür, dass die Dinge in Ordnung bleiben.Dictionary
s verwendet wird. Wenn Sie sich immer noch unwohl fühlen, können Sie es verwendenSortedDictionary
, aber meine eigene Erfahrung zeigt, dass dies unnötig ist.Wenn Sie die Zeitzone des Betrachters kennen, ist es möglicherweise klarer, Kalendertage auf der Tagesskala zu verwenden. Ich bin mit den .NET-Bibliotheken nicht vertraut, daher weiß ich leider nicht, wie Sie das in C # tun würden.
Auf Verbraucherseiten können Sie auch unter einer Minute von Hand wavier sein. "Vor weniger als einer Minute" oder "gerade jetzt" könnte gut genug sein.
quelle
Sie können dies versuchen. Ich denke, es wird richtig funktionieren.
quelle
Java für die clientseitige GWT-Nutzung:
quelle
@ Jeff
Doing eine Subtraktion auf
DateTime
kehrt einTimeSpan
ohnehin.Also kannst du es einfach tun
Ich bin auch überrascht zu sehen, wie die Konstanten von Hand multipliziert und dann Kommentare mit den Multiplikationen hinzugefügt werden. War das eine fehlgeleitete Optimierung?
quelle
Hier ist der Algorithmus, den der Stackoverflow verwendet, der jedoch im perlischen Pseudocode mit einer Fehlerbehebung (keine "vor einer Stunde") präziser umgeschrieben wurde. Die Funktion benötigt eine (positive) Anzahl von Sekunden und gibt eine menschenfreundliche Zeichenfolge wie "vor 3 Stunden" oder "gestern" zurück.
quelle
Sie können die TimeAgo-Erweiterung verwenden, die wie folgt aussieht:
Oder verwenden Sie das jQuery-Plugin mit der Razor-Erweiterung von Timeago.
quelle
Sie können die serverseitige Last reduzieren, indem Sie diese Logik clientseitig ausführen. Zeigen Sie die Quelle auf einigen Digg-Seiten als Referenz an. Der Server gibt einen Epochenzeitwert aus, der von Javascript verarbeitet wird. Auf diese Weise müssen Sie die Zeitzone des Endbenutzers nicht verwalten. Der neue serverseitige Code wäre ungefähr so:
Sie können dort sogar einen NOSCRIPT-Block hinzufügen und einfach einen ToString () ausführen.
quelle
Das habe ich von einem Blog von Bill Gates bekommen. Ich muss es in meinem Browserverlauf finden und gebe Ihnen den Link.
Der Javascript-Code, um dasselbe zu tun (wie angefordert):
Grundsätzlich arbeiten Sie in Sekunden ...
quelle
Ich denke, es gibt bereits eine Reihe von Antworten zu diesem Beitrag, aber man kann diese verwenden, die einfach wie das Plugin zu verwenden und auch für Programmierer leicht lesbar ist. Senden Sie Ihr bestimmtes Datum und erhalten Sie den Wert in Zeichenfolgenform:
quelle
quelle
Wenn Sie eine Ausgabe wie haben möchten
"2 days, 4 hours and 12 minutes ago"
, benötigen Sie eine Zeitspanne:Dann können Sie auf die gewünschten Werte zugreifen:
usw...
quelle
Ich würde hierfür einige praktische Erweiterungsmethoden bereitstellen und den Code lesbarer machen. Zunächst einige Erweiterungsmethoden für
Int32
.Dann eine für
DateTime
.Jetzt können Sie Folgendes tun:
quelle