सी # में सापेक्ष समय की गणना करें

एक विशिष्ट <�कोड> डेटटाइम मान दिया गया है, मैं सापेक्ष समय कैसे प्रदर्शित करूं, जैसे:

  • 2 घंटे पहले
  • 3 दिन पहले
  • एक महीने पहले
0
जोड़ा संपादित
विचारों: 17
क्या होगा यदि आप अब से भविष्य के लिए एक सापेक्ष समय की गणना करना चाहते हैं?
जोड़ा लेखक Jhonny D. Cano -Leftware-, स्रोत
.NET पैकेज github.com/NickStrupat/TimeAgo है जो पूछा जा रहा है कि काफी कुछ करता है।
जोड़ा लेखक Rossco, स्रोत
moment.js लाइब्रेरी लाइब्रेरी की एक बहुत अच्छी तारीख है .. आप अपनी जरूरतों के आधार पर उस (सर्वर साइड या क्लाइंट साइड) का उपयोग करने पर विचार कर सकते हैं। सिर्फ fyi क्योंकि कोई भी यहाँ इसका उल्लेख नहीं किया
जोड़ा लेखक code ninja, स्रोत

32 उत्तर

public static string RelativeDate(DateTime theDate)
{
    Dictionary thresholds = new Dictionary();
    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 = new TimeSpan((DateTime.Now.Ticks - theDate.Ticks));
            return string.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 "";
}

I prefer this version for its conciseness, and ability to add in new tick points. This could be encapsulated with a Latest() extension to Timespan instead of that long 1 liner, but for the sake of brevity in posting, this will do. This fixes the an hour ago, 1 hours ago, by providing an hour until 2 hours have elapsed

0
जोड़ा
मुझे इस फ़ंक्शन का उपयोग करके सभी प्रकार की समस्याएं मिल रही हैं, उदाहरण के लिए यदि आप 'theDate = dateTime.Now.AddMinutes (-40); मुझे '40 घंटे पहले 'मिल रहा है, लेकिन माइकल की रिफैक्टॉर्मोड प्रतिक्रिया के साथ, यह '40 मिनट पहले' सही हो जाता है?
जोड़ा लेखक GONeale, स्रोत
@CodeMonkeyKing द्वारा उल्लिखित समस्या को ठीक करने के लिए, आप एक सादे शब्दकोश के बजाय SortedDictionary का उपयोग कर सकते हैं: उपयोग समान है, लेकिन यह सुनिश्चित करता है कि चाबियाँ क्रमबद्ध हैं। लेकिन फिर भी, एल्गोरिदम में त्रुटियां हैं, क्योंकि RelativeDate (DateTime.Now.AddMonths (-3) .ddDays (-3)) वापस "95 महीने पहले" परवाह किए बिना आप किस शब्द प्रकार का उपयोग कर रहे हैं, जो गलत है (इसे "3 महीने पहले" या "4 महीने पहले" वापस लौटना चाहिए, जिस पर आप उपयोग कर रहे हैं) - भले ही -3 पिछले वर्ष में कोई तारीख नहीं बनाये ( मैंने दिसंबर
जोड़ा लेखक Matt, स्रोत
ओपी टी.डेज़> 30 भूल गया? टी। दिन / 30:
जोड़ा लेखक Lars Holm Jensen, स्रोत
हम्म, जबकि यह कोड काम कर सकता है, यह गलत है और यह मानने के लिए अमान्य है कि शब्दकोश में कुंजी का क्रम एक विशिष्ट क्रम में होगा। शब्दकोश ऑब्जेक्ट.GetHashCode() का उपयोग करता है जो लंबे समय तक वापस नहीं आता है लेकिन एक int! यदि आप इन्हें सॉर्ट करना चाहते हैं तो आपको सॉर्टेडलिस्ट का उपयोग करना चाहिए। यदि / /। / अन्य के सेट में थ्रेसहोल्ड का मूल्यांकन किया जा रहा है तो क्या गलत है? आपको तुलना की समान संख्या मिलती है। FYI लंबे समय तक हैश। मैक्सवेल्यू int.MinValue जैसा ही हो जाता है!
जोड़ा लेखक CodeMonkeyKing, स्रोत
मुझे लगता है कि आप शून्य खो रहे हैं, कोशिश करें: लंबे समय से = (DateTime.Now.Ticks - theDate.Ticks) / 10000000;
जोड़ा लेखक robnardo, स्रोत

यहां मैं यह कैसे करता हूं

var ts = new TimeSpan(DateTime.UtcNow.Ticks - dt.Ticks);
double delta = Math.Abs(ts.TotalSeconds);

if (delta < 60)
{
  return ts.Seconds == 1 ? "one second ago" : ts.Seconds + " seconds ago";
}
if (delta < 120)
{
  return "a minute ago";
}
if (delta < 2700) // 45 * 60
{
  return ts.Minutes + " minutes ago";
}
if (delta < 5400) // 90 * 60
{
  return "an hour ago";
}
if (delta < 86400) // 24 * 60 * 60
{
  return ts.Hours + " hours ago";
}
if (delta < 172800) // 48 * 60 * 60
{
  return "yesterday";
}
if (delta < 2592000) // 30 * 24 * 60 * 60
{
  return ts.Days + " days ago";
}
if (delta < 31104000) // 12 * 30 * 24 * 60 * 60
{
  int months = Convert.ToInt32(Math.Floor((double)ts.Days / 30));
  return months <= 1 ? "one month ago" : months + " months ago";
}
int years = Convert.ToInt32(Math.Floor((double)ts.Days / 365));
return years <= 1 ? "one year ago" : years + " years ago";

सुझाव? टिप्पणियाँ? इस एल्गोरिदम को सुधारने के तरीके?

0
जोड़ा
लेकिन वर्तमान में SO केवल 2 दिनों तक "टाइम पहले" प्रारूप दिखाएं। 2 दिनों से अधिक और आप मानक दिनांक समय प्रारूप डालते हैं। क्या मैं सही हू ?
जोड़ा लेखक Hugo Hilário, स्रोत
चूंकि उन सभी के बाद .....यह सिर्फ टाइमलाब हैं, आप बेहतर पठनीयता के लिए एक enum बना सकते हैं। enum TimeSlabs {सेकेंड = 60, मिनट = 60 * 60, घंटे = 60 * 60 * 24, दिन = 60 * 60 * 24 * 365,}
जोड़ा लेखक Antony Thomas, स्रोत
मुझे यह भी लगता है कि यह "कल" ​​की एक अजीब परिभाषा है, अगर मैं अब (रविवार शाम) शुक्रवार की शाम को "कल" ​​के रूप में एक उत्तर देखता हूं।
जोड़ा लेखक Paŭlo Ebermann, स्रोत
"<48 * 60 * 60s" "कल" ​​के लिए एक अपरंपरागत परिभाषा है। अगर बुधवार को 9 बजे है, तो क्या आप वास्तव में सोमवार को 9:01 बजे के बारे में सोचेंगे "कल"। मैंने सोचा था कि कल के लिए एल्गोरिदम या "एन दिन पहले" मध्यरात्रि के पहले / बाद में विचार करना चाहिए।
जोड़ा लेखक Joe, स्रोत
मैं बस var delta = ts. अवधि() असाइन करता हूं, इसलिए आपको TimeSpan ऑब्जेक्ट में पूर्ण मान मिलता है, और फिर आप if (डेल्टा) जैसी सामग्री कर सकते हैं या अगर (डेल्टा <�टाइमस्पेन। फ्रॉमडेज़ (1)) । अभी भी पठनीय, और गुप्त संख्या या स्थिरांक की कोई आवश्यकता नहीं है।
जोड़ा लेखक Rufus L, स्रोत
कंपाइलर्स आमतौर पर 24 * 60 * 60 की तरह पूर्व-गणना निरंतर अभिव्यक्तियों पर बहुत अच्छे होते हैं, ताकि आप सीधे इसे 86400 होने और टिप्पणियों में मूल अभिव्यक्ति को रखने के बजाय उन लोगों का उपयोग कर सकें
जोड़ा लेखक zvolkov, स्रोत
@bzlm मुझे लगता है कि मैंने एक परियोजना के लिए किया था जिस पर मैं काम कर रहा था। यहां मेरी प्रेरणा दूसरों को सतर्क करना था कि इस कोड नमूने से सप्ताह छोड़े गए थे। ऐसा करने के लिए, यह मेरे लिए बहुत सीधे लग रहा था।
जोड़ा लेखक jray, स्रोत
ध्यान दिया कि इस समारोह में हफ्तों को शामिल नहीं किया गया है
जोड़ा लेखक jray, स्रोत
मुझे लगता है कि एल्गोरिदम सुधारने का अच्छा तरीका सटीकता बढ़ाने के लिए "2 महीने 21 दिन पूर्व", "1 घंटा 40 मिनट पहले" जैसी 2 इकाइयों को प्रदर्शित कर रहा है।
जोड़ा लेखक Evgeny Levin, स्रोत
@ जेफी, आप लीप वर्ष और संबंधित चेक के लिए गणना चूक गए
जोड़ा लेखक Saboor Awan, स्रोत
45 मिनट "एक घंटा पहले" के रूप में प्रदर्शित किया जा रहा है ?! क्या यह एक बग नहीं है? यदि ऐसा नहीं है, तो सेकंड और मिनट के लिए ऐसा क्यों न करें?
जोड़ा लेखक Eldritch Conundrum, स्रोत

@jeff

IMHO आपका थोड़ा लंबा लगता है। हालांकि यह "कल" ​​और "वर्षों" के समर्थन के साथ थोड़ा और मजबूत लगता है। लेकिन मेरे अनुभव में जब इसका उपयोग किया जाता है तो व्यक्ति को पहले 30 दिनों में सामग्री देखने की संभावना है। यह केवल वास्तव में कट्टर लोग हैं जो उसके बाद आते हैं। इसलिए मैं आमतौर पर यह छोटा और सरल रखने का चुनाव करता हूं।

यह वह तरीका है जिसका मैं वर्तमान में अपनी वेबसाइटों में से एक पर उपयोग कर रहा हूं। यह केवल एक सापेक्ष दिन, घंटे, समय देता है। और फिर उपयोगकर्ता को आउटपुट में "पहले" पर थप्पड़ मारना पड़ता है।

public static string ToLongString(this TimeSpan time)
{
    string output = String.Empty;

    if (time.Days > 0)
        output += time.Days + " days ";

    if ((time.Days == 0 || time.Days == 1) && time.Hours > 0)
        output += time.Hours + " hr ";

    if (time.Days == 0 && time.Minutes > 0)
        output += time.Minutes + " min ";

    if (output.Length == 0)
        output += time.Seconds + " sec";

    return output.Trim();
}
0
जोड़ा

जेफ, आपका कोड अच्छा है लेकिन स्थिरांक के साथ स्पष्ट हो सकता है (कोड पूर्ण में सुझाए गए अनुसार)।

const int SECOND = 1;
const int MINUTE = 60 * SECOND;
const int HOUR = 60 * MINUTE;
const int DAY = 24 * HOUR;
const int MONTH = 30 * DAY;

var ts = new TimeSpan(DateTime.UtcNow.Ticks - yourDate.Ticks);
double delta = Math.Abs(ts.TotalSeconds);

if (delta < 1 * MINUTE)
  return ts.Seconds == 1 ? "one second ago" : ts.Seconds + " seconds ago";

if (delta < 2 * MINUTE)
  return "a minute ago";

if (delta < 45 * MINUTE)
  return ts.Minutes + " minutes ago";

if (delta < 90 * MINUTE)
  return "an hour ago";

if (delta < 24 * HOUR)
  return ts.Hours + " hours ago";

if (delta < 48 * HOUR)
  return "yesterday";

if (delta < 30 * DAY)
  return ts.Days + " days ago";

if (delta < 12 * MONTH)
{
  int months = Convert.ToInt32(Math.Floor((double)ts.Days / 30));
  return months <= 1 ? "one month ago" : months + " months ago";
}
else
{
  int years = Convert.ToInt32(Math.Floor((double)ts.Days / 365));
  return years <= 1 ? "one year ago" : years + " years ago";
}
0
जोड़ा
@ एटोमो: नहीं, यह संभव नहीं है
जोड़ा लेखक André Pena, स्रोत
मैं वास्तव में समझ में नहीं आता कि लोग कुल मिनट, कुल दिन, कुलहोर्स का उपयोग क्यों नहीं करते हैं ... इससे यह बहुत कम हो जाएगा ...
जोड़ा लेखक lord.fist, स्रोत
क्या किसी के पास कोई कस्टम प्रारूप स्ट्रिंग के साथ इसे कार्यान्वित किया जा सकता है, इस पर कोई मार्गदर्शन है, ताकि 'string.Format (mydate, "dd / MM / yyyy (YTT)") == "26/04/2011 (आज)"?
जोड़ा लेखक Neil Barnwell, स्रोत
मुझे जुनून के साथ ऐसे स्थिरांक से नफरत है। क्या यह किसी के लिए गलत दिखता है? <�कोड> थ्रेड। सो जाओ (1 * मिनट) ? क्योंकि यह 1000 के कारक से गलत है।
जोड़ा लेखक Roman Starkov, स्रोत
एक और बग: क्योंकि डेल्टा पूर्ण है, "अभी तक नहीं" का मार्ग कभी नहीं पहुंचा है। में बदलें: डेल्टा = ts कुल सेकेंड;
जोड़ा लेखक StefanG, स्रोत
मुझे लगता है कि यदि स्थिरांक का नाम बदलकर उनके मूल्य में सटीक रूप से वर्णन किया गया है, तो यह समझना आसान होगा। तो SecondsPerMinute = 60; MinutesPerHour = 60; SecondsPerHour = मिनट PerHour * SecondsPerHour; इत्यादि। बस इसे MINUTE = 60 कहकर पाठक को यह निर्धारित करने की अनुमति नहीं है कि मूल्य क्या है।
जोड़ा लेखक slolife, स्रोत
गड़बड़ी "कल" ​​में है, यह कुछ सटीक नहीं है क्योंकि मैं आज सुबह 7 बजे देख रहा हूं, जो कि दो दिन पहले 10 बजे हुआ था, "कल" ​​पैदा करेगा, जो सटीक नहीं है।
जोड़ा लेखक Ayyash, स्रोत
कुछ भाषाओं में आप स्विच() - कथन, जैसे>,> =, <= आदि के लिए किसी भी प्रकार की तुलना ऑपरेटर निर्दिष्ट कर सकते हैं ... क्या यह सी # में किसी भी तरह से संभव नहीं है?
जोड़ा लेखक Atmocreations, स्रोत
क्यों नहीं var ts = DateTime.UtcNow - yourDate; , यह आसान है? Math.Abs ​​ के लिए औचित्य क्या है?
जोड़ा लेखक Jeppe Stig Nielsen, स्रोत
अगर नहीं होना चाहिए (डेल्टा <= 90 * मिनट)? किसी तरह मुझे "1 घंटे पहले" मिला।
जोड़ा लेखक Roberto, स्रोत
गलत 'कल' या 'दिन पहले' मूल्य के बारे में कोई भी क्यों नहीं (जो को छोड़कर) ??? कल एक घंटे की गणना नहीं है, लेकिन दिन-प्रतिदिन की गणना है। तो हाँ, यह कम से कम दो लगातार मामले में एक गलत कोड है।
जोड़ा लेखक CtrlX, स्रोत
const int SECOND = 1; तो अजीब एक सेकंड एक सेकंड है।
जोड़ा लेखक seriousdev, स्रोत
इस प्रकार का कोड स्थानीयकरण के लिए लगभग असंभव है। यदि आपके ऐप को केवल अंग्रेजी में रहना है, तो ठीक है। लेकिन अगर आप अन्य भाषाओं में कूदते हैं, तो आप इस तरह तर्क करने के लिए खुद से नफरत करेंगे। बस इतना पता है ...
जोड़ा लेखक Nik Reiman, स्रोत
@romkyns थ्रेड। नींद (1 * मिनट) करता है मुझे गलत लग रहा है, और ऐसा इसलिए है क्योंकि MINUTE में कोई इकाई निर्दिष्ट नहीं है। अब, MINUTE_IN_SECONDS , जो स्पॉट और आसानी से तय करना आसान होगा: थ्रेड। सोएं (1 * MINUTE_IN_SECONDS * 1000) या थ्रेड। नींद (1 * MINUTE_IN_SECONDS * SECOND_IN_MILLISECONDS) । (वर्बोज़ प्राप्त करना, लेकिन छिपी हुई बग से बेहतर।)
जोड़ा लेखक jpmc26, स्रोत
जब मैं सभी पूंजी देखता हूं तो मुझे सी / सी ++ के लिए विकसित कोडिंग शैलियों पर वापस लगता है। माइक्रोसॉफ्ट की सिफारिश पास्कल का पीछा किया गया है। इस प्रश्न का उत्तर देखें - constants"> stackoverflow.com के लिए
जोड़ा लेखक CodeMonkeyKing, स्रोत
@nick यदि आप इसे एक IValueConverter में रखते हैं, तो यह आपके लिए स्वचालित रूप से CultureInfo को संभालेगा (माना जाता है कि आपने पहले संस्कृति सेट की है)
जोड़ा लेखक Lance McCarthy, स्रोत
आप डेटटाइम प्रकार पर परिभाषित कॉन्स का उपयोग करके समाधान को अनुकूलित कर सकते हैं, आपको उन्हें मैन्युअल रूप से परिभाषित नहीं करना है।
जोड़ा लेखक SOAL ABDELDJALLIL, स्रोत

मैंने सोचा कि मैं इसे कक्षाओं और बहुरूपता का उपयोग करके एक शॉट दूंगा। मेरे पास पिछली पुनरावृत्ति थी जो उप-वर्गीकरण का उपयोग करती थी जो बहुत अधिक ओवरहेड के रास्ते में समाप्त होती थी। मैंने एक अधिक लचीला प्रतिनिधि / सार्वजनिक संपत्ति ऑब्जेक्ट मॉडल पर स्विच किया है जो काफी बेहतर है। मेरा कोड बहुत थोड़ा सटीक है, मेरी इच्छा है कि मैं "महीने पहले" उत्पन्न करने के लिए एक बेहतर तरीके से आ सकता हूं जो बहुत अधिक इंजीनियर नहीं लग रहा था।

मुझे लगता है कि मैं अभी भी जेफ के अगर-कैस्केड के साथ रहूंगा क्योंकि यह कम कोड है और यह आसान है (यह सुनिश्चित करना निश्चित रूप से आसान है कि यह अपेक्षित काम करेगा)।

नीचे दिए गए कोड के लिए PrintRelativeTime.GetRelativeTimeMessage (टाइमस्पेन पहले) सापेक्ष समय संदेश (उदा। "कल") देता है।

public class RelativeTimeRange : IComparable
{
    public TimeSpan UpperBound { get; set; }

    public delegate string RelativeTimeTextDelegate(TimeSpan timeDelta);

    public RelativeTimeTextDelegate MessageCreator { get; set; }

    public int CompareTo(object obj)
    {
        if (!(obj is RelativeTimeRange))
        {
            return 1;
        }
        // note that this sorts in reverse order to the way you'd expect, 
        // this saves having to reverse a list later
        return (obj as RelativeTimeRange).UpperBound.CompareTo(UpperBound);
    }
}

public class PrintRelativeTime
{
    private static List timeRanges;

    static PrintRelativeTime()
    {
        timeRanges = new List{
            new RelativeTimeRange
            {
                UpperBound = TimeSpan.FromSeconds(1),
                MessageCreator = (delta) => 
                { return "one second ago"; }
            }, 
            new RelativeTimeRange
            {
                UpperBound = TimeSpan.FromSeconds(60),
                MessageCreator = (delta) => 
                { return delta.Seconds + " seconds ago"; }

            }, 
            new RelativeTimeRange
            {
                UpperBound = TimeSpan.FromMinutes(2),
                MessageCreator = (delta) => 
                { return "one minute ago"; }
            }, 
            new RelativeTimeRange
            {
                UpperBound = TimeSpan.FromMinutes(60),
                MessageCreator = (delta) => 
                { return delta.Minutes + " minutes ago"; }
            }, 
            new RelativeTimeRange
            {
                UpperBound = TimeSpan.FromHours(2),
                MessageCreator = (delta) => 
                { return "one hour ago"; }
            }, 
            new RelativeTimeRange
            {
                UpperBound = TimeSpan.FromHours(24),
                MessageCreator = (delta) => 
                { return delta.Hours + " hours ago"; }
            }, 
            new RelativeTimeRange
            {
                UpperBound = TimeSpan.FromDays(2),
                MessageCreator = (delta) => 
                { return "yesterday"; }
            }, 
            new RelativeTimeRange
            {
                UpperBound = DateTime.Now.Subtract(DateTime.Now.AddMonths(-1)),
                MessageCreator = (delta) => 
                { return delta.Days + " days ago"; }
            }, 
            new RelativeTimeRange
            {
                UpperBound = DateTime.Now.Subtract(DateTime.Now.AddMonths(-2)),
                MessageCreator = (delta) => 
                { return "one month ago"; }
            }, 
            new RelativeTimeRange
            {
                UpperBound = DateTime.Now.Subtract(DateTime.Now.AddYears(-1)),
                MessageCreator = (delta) => 
                { return (int)Math.Floor(delta.TotalDays / 30) + " months ago"; }
            }, 
            new RelativeTimeRange
            {
                UpperBound = DateTime.Now.Subtract(DateTime.Now.AddYears(-2)),
                MessageCreator = (delta) => 
                { return "one year ago"; }
            }, 
            new RelativeTimeRange
            {
                UpperBound = TimeSpan.MaxValue,
                MessageCreator = (delta) => 
                { return (int)Math.Floor(delta.TotalDays / 365.24D) + " years ago"; }
            }
        };

        timeRanges.Sort();
    }

    public static string GetRelativeTimeMessage(TimeSpan ago)
    {
        RelativeTimeRange postRelativeDateRange = timeRanges[0];

        foreach (var timeRange in timeRanges)
        {
            if (ago.CompareTo(timeRange.UpperBound) <= 0)
            {
                postRelativeDateRange = timeRange;
            }
        }

        return postRelativeDateRange.MessageCreator(ago);
    }
}
0
जोड़ा

मैं इसे क्लाइंट पक्ष पर भी कंप्यूटिंग करने की अनुशंसा करता हूं। सर्वर के लिए कम काम।

निम्नलिखित संस्करण है जिसका मैं उपयोग करता हूं (जैच लेदरमैन से)

/*
 * Javascript Humane Dates
 * Copyright (c) 2008 Dean Landolt (deanlandolt.com)
 * Re-write by Zach Leatherman (zachleat.com)
 * 
 * Adopted from the John Resig's pretty.js
 * at http://ejohn.org/blog/javascript-pretty-date
 * and henrah's proposed modification 
 * at http://ejohn.org/blog/javascript-pretty-date/#comment-297458
 * 
 * Licensed under the MIT license.
 */

function humane_date(date_str){
        var time_formats = [
                [60, 'just now'],
                [90, '1 minute'], // 60*1.5
                [3600, 'minutes', 60], // 60*60, 60
                [5400, '1 hour'], // 60*60*1.5
                [86400, 'hours', 3600], // 60*60*24, 60*60
                [129600, '1 day'], // 60*60*24*1.5
                [604800, 'days', 86400], // 60*60*24*7, 60*60*24
                [907200, '1 week'], // 60*60*24*7*1.5
                [2628000, 'weeks', 604800], // 60*60*24*(365/12), 60*60*24*7
                [3942000, '1 month'], // 60*60*24*(365/12)*1.5
                [31536000, 'months', 2628000], // 60*60*24*365, 60*60*24*(365/12)
                [47304000, '1 year'], // 60*60*24*365*1.5
                [3153600000, 'years', 31536000], // 60*60*24*365*100, 60*60*24*365
                [4730400000, '1 century'] // 60*60*24*365*100*1.5
        ];

        var time = ('' + date_str).replace(/-/g,"/").replace(/[TZ]/g," "),
                dt = new Date,
                seconds = ((dt - new Date(time) + (dt.getTimezoneOffset() * 60000)) / 1000),
                token = ' ago',
                i = 0,
                format;

        if (seconds < 0) {
                seconds = Math.abs(seconds);
                token = '';
        }

        while (format = time_formats[i++]) {
                if (seconds < format[0]) {
                        if (format.length == 2) {
                                return format[1] + (i > 1 ? token : ''); // Conditional so we don't return Just Now Ago
                        } else {
                                return Math.round(seconds / format[2]) + ' ' + format[1] + (i > 1 ? token : '');
                        }
                }
        }

        // overflow for centuries
        if(seconds > 4730400000)
                return Math.round(seconds / 4730400000) + ' centuries' + token;

        return date_str;
};

if(typeof jQuery != 'undefined') {
        jQuery.fn.humane_dates = function(){
                return this.each(function(){
                        var date = humane_date(this.title);
                        if(date && jQuery(this).text() != date) // don't modify the dom if we don't have to
                                jQuery(this).text(date);
                });
        };
}
0
जोड़ा
प्रश्न सी # टैग किया गया क्यों जावास्क्रिप्ट कोड ?
जोड़ा लेखक Kiquenet, स्रोत

PHP में, मैं इसे इस तरह से करता हूं:

<?php
function timesince($original) {
    // array of time period chunks
    $chunks = array(
        array(60 * 60 * 24 * 365 , 'year'),
        array(60 * 60 * 24 * 30 , 'month'),
        array(60 * 60 * 24 * 7, 'week'),
        array(60 * 60 * 24 , 'day'),
        array(60 * 60 , 'hour'),
        array(60 , 'minute'),
    );

    $today = time(); /* Current unix time  */
    $since = $today - $original;

    if($since > 604800) {
    $print = date("M jS", $original);

    if($since > 31536000) {
        $print .= ", " . date("Y", $original);
    }

    return $print;
}

// $j saves performing the count function each time around the loop
for ($i = 0, $j = count($chunks); $i < $j; $i++) {

    $seconds = $chunks[$i][0];
    $name = $chunks[$i][1];

    // finding the biggest chunk (if the chunk fits, break)
    if (($count = floor($since / $seconds)) != 0) {
        break;
    }
}

$print = ($count == 1) ? '1 '.$name : "$count {$name}s";

return $print . " ago";

} ?>
0
जोड़ा
प्रश्न सी # टैग किया गया है। यह PHP कोड क्यों? आईएमएचओ, केवल सी # कोड लागू होता है
जोड़ा लेखक Kiquenet, स्रोत

@Jeff

<�पी> var ts = नया   टाइमस्पेन (DateTime.UtcNow.Ticks -   dt.Ticks);

डेटटाइम पर एक घटाव करने से वैसे भी एक TimeSpan लौटाता है।

तो आप बस कर सकते हैं

(DateTime.UtcNow - dt).TotalSeconds

मैं हाथ से गुणा करके स्थिरांक को देखकर आश्चर्यचकित हूं और फिर गुणाओं के साथ टिप्पणियों को जोड़ा गया। क्या यह कुछ गुमराह अनुकूलन था?

0
जोड़ा

यह, मैं बिल गेट्स ब्लॉग में से एक से मिला। मुझे इसे अपने ब्राउज़र इतिहास पर ढूंढना होगा और मैं आपको लिंक दूंगा।

जावास्क्रिप्ट कोड एक ही काम करने के लिए (अनुरोध के रूप में):

function posted(t) {
    var now = new Date();
    var diff = parseInt((now.getTime() - Date.parse(t)) / 1000);
    if (diff < 60) { return 'less than a minute ago'; }
    else if (diff < 120) { return 'about a minute ago'; }
    else if (diff < (2700)) { return (parseInt(diff / 60)).toString() + ' minutes ago'; }
    else if (diff < (5400)) { return 'about an hour ago'; }
    else if (diff < (86400)) { return 'about ' + (parseInt(diff / 3600)).toString() + ' hours ago'; }
    else if (diff < (172800)) { return '1 day ago'; } 
    else {return (parseInt(diff / 86400)).toString() + ' days ago'; }
}

असल में, आप सेकंड के मामले में काम करते हैं ...

0
जोड़ा

आप TimeAgo एक्सटेंशन का उपयोग कर सकते हैं, जिससे निम्न जैसा दिखता है:

public static string TimeAgo(this DateTime dateTime)
{
    string result = string.Empty;
    var timeSpan = DateTime.Now.Subtract(dateTime);

    if (timeSpan <= TimeSpan.FromSeconds(60))
    {
        result = string.Format("{0} seconds ago", timeSpan.Seconds);
    }
    else if (timeSpan <= TimeSpan.FromMinutes(60))
    {
        result = timeSpan.Minutes > 1 ? 
            String.Format("about {0} minutes ago", timeSpan.Minutes) :
            "about a minute ago";
    }
    else if (timeSpan <= TimeSpan.FromHours(24))
    {
        result = timeSpan.Hours > 1 ? 
            String.Format("about {0} hours ago", timeSpan.Hours) : 
            "about an hour ago";
    }
    else if (timeSpan <= TimeSpan.FromDays(30))
    {
        result = timeSpan.Days > 1 ? 
            String.Format("about {0} days ago", timeSpan.Days) : 
            "yesterday";
    }
    else if (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;
}

या टाइमगो से रेजर एक्सटेंशन के साथ jQuery प्लगइन का उपयोग करें।

0
जोड़ा

मैं इसके लिए कुछ आसान एक्सटेंशन विधियां प्रदान करूंगा और कोड को और अधिक पठनीय बना दूंगा। सबसे पहले, Int32 के लिए एक्सटेंशन विधियों में से कुछ।

public static class TimeSpanExtensions {

    public static TimeSpan Days(this int value) {

        return new TimeSpan(value, 0, 0, 0);
    }

    public static TimeSpan Hours(this int value) {

        return new TimeSpan(0, value, 0, 0);
    }

    public static TimeSpan Minutes(this int value) {

        return new TimeSpan(0, 0, value, 0);
    }

    public static TimeSpan Seconds(this int value) {

        return new TimeSpan(0, 0, 0, value);
    }

    public static TimeSpan Milliseconds(this int value) {

        return new TimeSpan(0, 0, 0, 0, value);
    }

    public static DateTime Ago(this TimeSpan value) {

        return DateTime.Now - value;
    }
}

फिर, डेटटाइम के लिए एक।

public static class DateTimeExtensions {

    public static DateTime Ago(this DateTime dateTime, TimeSpan delta) {

        return dateTime - delta;
    }
}

अब, आप नीचे कुछ ऐसा कर सकते हैं:

var date = DateTime.Now;
date.Ago(2.Days()); // 2 days ago
date.Ago(7.Hours()); // 7 hours ago
date.Ago(567.Milliseconds()); // 567 milliseconds ago
0
जोड़ा

यहां PHP के लिए जेफ्स स्क्रिप्ट से पुनः लिखें:

define("SECOND", 1);
define("MINUTE", 60 * SECOND);
define("HOUR", 60 * MINUTE);
define("DAY", 24 * HOUR);
define("MONTH", 30 * DAY);
function relativeTime($time)
{   
    $delta = time() - $time;

    if ($delta < 1 * MINUTE)
    {
        return $delta == 1 ? "one second ago" : $delta . " seconds ago";
    }
    if ($delta < 2 * MINUTE)
    {
      return "a minute ago";
    }
    if ($delta < 45 * MINUTE)
    {
        return floor($delta / MINUTE) . " minutes ago";
    }
    if ($delta < 90 * MINUTE)
    {
      return "an hour ago";
    }
    if ($delta < 24 * HOUR)
    {
      return floor($delta / HOUR) . " hours ago";
    }
    if ($delta < 48 * HOUR)
    {
      return "yesterday";
    }
    if ($delta < 30 * DAY)
    {
        return floor($delta / DAY) . " days ago";
    }
    if ($delta < 12 * MONTH)
    {
      $months = floor($delta / DAY / 30);
      return $months <= 1 ? "one month ago" : $months . " months ago";
    }
    else
    {
        $years = floor($delta / DAY / 365);
        return $years <= 1 ? "one year ago" : $years . " years ago";
    }
}    
0
जोड़ा
प्रश्न सी # टैग किया गया क्यों PHP कोड ?
जोड़ा लेखक Kiquenet, स्रोत

using Fluent DateTime https://github.com/FluentDateTime

var dateTime1 = 2.Hours().Ago();
var dateTime2 = 3.Days().Ago();
var dateTime3 = 1.Months().Ago();
var dateTime4 = 5.Hours().FromNow();
var dateTime5 = 2.Weeks().FromNow();
var dateTime6 = 40.Seconds().FromNow();
0
जोड़ा

यहां एल्गोरिदम स्टैकओवरफ्लो का उपयोग किया गया है लेकिन एक बग फिक्स के साथ पर्लोड स्यूडोकोड में अधिक संक्षेप में फिर से लिखा गया है (नहीं "एक घंटा पहले")। फ़ंक्शन कुछ सेकंड पहले (पॉजिटिव) संख्या लेता है और "3 घंटे पहले" या "कल" ​​जैसे मानव-अनुकूल स्ट्रिंग देता है।

agoify($delta)
  local($y, $mo, $d, $h, $m, $s);
  $s = floor($delta);
  if($s<=1)            return "a second ago";
  if($s<60)            return "$s seconds ago";
  $m = floor($s/60);
  if($m==1)            return "a minute ago";
  if($m<45)            return "$m minutes ago";
  $h = floor($m/60);
  if($h==1)            return "an hour ago";
  if($h<24)            return "$h hours ago";
  $d = floor($h/24);
  if($d<2)             return "yesterday";
  if($d<30)            return "$d days ago";
  $mo = floor($d/30);
  if($mo<=1)           return "a month ago";
  $y = floor($mo/12);
  if($y<1)             return "$mo months ago";
  if($y==1)            return "a year ago";
  return "$y years ago";
0
जोड़ा
using System;
using System.Collections.Generic;
using System.Linq;

public static class RelativeDateHelper
{
    private static Dictionary> sm_Dict = null;

    private static Dictionary> DictionarySetup()
    {
        var dict = new Dictionary>();
        dict.Add(0.75, (mins) => "less than a minute");
        dict.Add(1.5, (mins) => "about a minute");
        dict.Add(45, (mins) => string.Format("{0} minutes", Math.Round(mins)));
        dict.Add(90, (mins) => "about an hour");
        dict.Add(1440, (mins) => string.Format("about {0} hours", Math.Round(Math.Abs(mins / 60)))); // 60 * 24
        dict.Add(2880, (mins) => "a day"); // 60 * 48
        dict.Add(43200, (mins) => string.Format("{0} days", Math.Floor(Math.Abs(mins / 1440)))); // 60 * 24 * 30
        dict.Add(86400, (mins) => "about a month"); // 60 * 24 * 60
        dict.Add(525600, (mins) => string.Format("{0} months", Math.Floor(Math.Abs(mins / 43200)))); // 60 * 24 * 365 
        dict.Add(1051200, (mins) => "about a year"); // 60 * 24 * 365 * 2
        dict.Add(double.MaxValue, (mins) => string.Format("{0} years", Math.Floor(Math.Abs(mins / 525600))));

        return dict;
    }

    public static string ToRelativeDate(this DateTime input)
    {
        TimeSpan oSpan = DateTime.Now.Subtract(input);
        double TotalMinutes = oSpan.TotalMinutes;
        string Suffix = " ago";

        if (TotalMinutes < 0.0)
        {
            TotalMinutes = Math.Abs(TotalMinutes);
            Suffix = " from now";
        }

        if (null == sm_Dict)
            sm_Dict = DictionarySetup();

        return sm_Dict.First(n => TotalMinutes < n.Key).Value.Invoke(TotalMinutes) + Suffix;
    }
}

इस प्रश्न का एक और उत्तर जैसा ही एक एक्सटेंशन के रूप में एक स्थिर शब्दकोश के साथ विधि।

0
जोड़ा
शब्दकोश आपको यहां क्या खरीदता है?
जोड़ा लेखक StriplingWarrior, स्रोत
समझा। मैं बस सोच रहा था, क्योंकि Dictionary पर प्रलेखन बताता है कि "ऑर्डर जिसमें आइटम लौटाए गए हैं, अनिर्धारित है," ( msdn.microsoft.com/en-us/library/xfhwa508.aspx ) शायद जब आप परवाह नहीं करते हैं तो इसका उपयोग करने के लिए सबसे अच्छी डेटा संरचना नहीं है लुकअप के समय के बारे में चीजें रखने के क्रम में।
जोड़ा लेखक StriplingWarrior, स्रोत
StriplingWarrior: एक स्विच स्टेटमेंट की तुलना में पढ़ने और संशोधित करने की आसानी या if / else बयान के ढेर। शब्दकोश का स्थैतिक अर्थ यह है कि यह और Func <,> ऑब्जेक्ट्स को हर बार बनाया जाना नहीं है जब हम ToRelativeDate का उपयोग करना चाहते हैं; यह मेरे उत्तर में लिंक किए गए एक की तुलना में केवल एक बार बनाया गया है।
जोड़ा लेखक Chris Charabaruk, स्रोत
StriplingWarrior: मेरा मानना ​​है कि LINQ शब्दकोश s के साथ उपयोग किए जाने पर इसे ध्यान में रखता है। यदि आप अभी भी इसके साथ असहज हैं, तो आप SortedDictionary </ कोड> , लेकिन मेरा खुद का अनुभव दिखाता है कि अनावश्यक होना।
जोड़ा लेखक Chris Charabaruk, स्रोत
public string getRelativeDateTime(DateTime date)
{
    TimeSpan ts = DateTime.Now - date;
    if (ts.TotalMinutes < 1)//seconds ago
        return "just now";
    if (ts.TotalHours < 1)//min ago
        return (int)ts.TotalMinutes == 1 ? "1 Minute ago" : (int)ts.TotalMinutes + " Minutes ago";
    if (ts.TotalDays < 1)//hours ago
        return (int)ts.TotalHours == 1 ? "1 Hour ago" : (int)ts.TotalHours + " Hours ago";
    if (ts.TotalDays < 7)//days ago
        return (int)ts.TotalDays == 1 ? "1 Day ago" : (int)ts.TotalDays + " Days ago";
    if (ts.TotalDays < 30.4368)//weeks ago
        return (int)(ts.TotalDays / 7) == 1 ? "1 Week ago" : (int)(ts.TotalDays / 7) + " Weeks ago";
    if (ts.TotalDays < 365.242)//months ago
        return (int)(ts.TotalDays / 30.4368) == 1 ? "1 Month ago" : (int)(ts.TotalDays / 30.4368) + " Months ago";
    //years ago
    return (int)(ts.TotalDays / 365.242) == 1 ? "1 Year ago" : (int)(ts.TotalDays / 365.242) + " Years ago";
}

एक महीने और साल के दिनों के लिए रूपांतरण मान Google से लिया गया था।

0
जोड़ा

पार्टी के कुछ देर बाद, लेकिन मुझे पिछले और भविष्य दोनों तिथियों के लिए ऐसा करने की आवश्यकता थी, इसलिए मैंने जेफ और विन्सेंट की इसमें। यह एक ternarytastic extravaganza है! :)

public static class DateTimeHelper
    {
        private const int SECOND = 1;
        private const int MINUTE = 60 * SECOND;
        private const int HOUR = 60 * MINUTE;
        private const int DAY = 24 * HOUR;
        private const int MONTH = 30 * DAY;

        /// 
/// Returns a friendly version of the provided DateTime, relative to now. E.g.: "2 days ago", or "in 6 months". ///
 
        /// 
The DateTime to compare to Now
        /// A friendly string
        public static string GetFriendlyRelativeTime(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 ? new TimeSpan(dateTime.Ticks - DateTime.UtcNow.Ticks) : new TimeSpan(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";
            }
        }
    }
0
जोड़ा

मुझे लगता है कि इस पोस्ट से संबंधित कई जवाब पहले से ही हैं, लेकिन कोई इसका उपयोग कर सकता है जो प्लगइन की तरह उपयोग करना आसान है और प्रोग्रामर के लिए आसानी से पठनीय भी है। अपनी विशिष्ट तिथि भेजें, और स्ट्रिंग फॉर्म में अपना मान प्राप्त करें:

public string RelativeDateTimeCount(DateTime inputDateTime)
{
    string outputDateTime = string.Empty;
    TimeSpan ts = DateTime.Now - inputDateTime;

    if (ts.Days > 7)
    { outputDateTime = inputDateTime.ToString("MMMM d, yyyy"); }

    else if (ts.Days > 0)
    {
        outputDateTime = ts.Days == 1 ? ("about 1 Day ago") : ("about " + ts.Days.ToString() + " Days ago");
    }
    else if (ts.Hours > 0)
    {
        outputDateTime = ts.Hours == 1 ? ("an hour ago") : (ts.Hours.ToString() + " hours ago");
    }
    else if (ts.Minutes > 0)
    {
        outputDateTime = ts.Minutes == 1 ? ("1 minute ago") : (ts.Minutes.ToString() + " minutes ago");
    }
    else outputDateTime = "few seconds ago";

    return outputDateTime;
}
0
जोड़ा

मेरा रास्ता बहुत आसान है। आप चाहें तो रिटर्न स्ट्रिंग्स के साथ ट्विक कर सकते हैं

    public static string TimeLeft(DateTime utcDate)
    {
        TimeSpan timeLeft = DateTime.UtcNow - utcDate;
        string timeLeftString = "";
        if (timeLeft.Days > 0)
        {
            timeLeftString += timeLeft.Days == 1 ? timeLeft.Days + " day" : timeLeft.Days + " days";
        }
        else if (timeLeft.Hours > 0)
        {
            timeLeftString += timeLeft.Hours == 1 ? timeLeft.Hours + " hour" : timeLeft.Hours + " hours";
        }
        else
        {
            timeLeftString += timeLeft.Minutes == 1 ? timeLeft.Minutes+" minute" : timeLeft.Minutes + " minutes";
        }
        return timeLeftString;
    }
0
जोड़ा

निश्चित रूप से '1 घंटे पहले' समस्या से छुटकारा पाने के लिए एक आसान फिक्स उस विंडो को बढ़ाने के लिए होगा जो 'एक घंटा पहले' मान्य है। परिवर्तन

if (delta < 5400) // 90 * 60
{
    return "an hour ago";
}

में

if (delta < 7200) // 120 * 60
{
    return "an hour ago";
}

इसका मतलब है कि 110 मिनट पहले जो कुछ हुआ वह 'एक घंटा पहले' के रूप में पढ़ा जाएगा - यह सही नहीं हो सकता है, लेकिन मैं कहूंगा कि यह '1 घंटे पहले' की वर्तमान स्थिति से बेहतर है।

0
जोड़ा

दुनिया और उसके पति को कोड नमूने पोस्ट करने लगते हैं, इन कुछ उत्तरों के आधार पर मैंने कुछ समय पहले लिखा था।

मुझे इस कोड के लिए स्थानीयकरण योग्य होने की एक विशिष्ट आवश्यकता थी। तो मेरे पास दो वर्ग हैं? व्याकरण , जो स्थानीयकरण योग्य शर्तों को निर्दिष्ट करता है, और FuzzyDateExtensions , जिसमें विस्तार विधियों का एक समूह होता है। मुझे भविष्य के समय से निपटने की कोई जरूरत नहीं थी, इसलिए इस कोड के साथ उन्हें संभालने के लिए कोई प्रयास नहीं किया गया।

मैंने स्रोत में कुछ XMLdoc छोड़ा है, लेकिन ब्रेवटी के लिए सबसे अधिक हटा दिया गया है (जहां वे स्पष्ट होंगे)। मैंने यहां हर वर्ग के सदस्य को भी शामिल नहीं किया है:

public class Grammar
{
    /// 
Gets or sets the term for "just now".
 
    public string JustNow { get; set; }
    /// 
Gets or sets the term for "X minutes ago".
 
    /// 
    ///     This is a  pattern, where {0}
    ///     is the number of minutes.
    /// 
    public string MinutesAgo { get; set; }
    public string OneHourAgo { get; set; }
    public string HoursAgo { get; set; }
    public string Yesterday { get; set; }
    public string DaysAgo { get; set; }
    public string LastMonth { get; set; }
    public string MonthsAgo { get; set; }
    public string LastYear { get; set; }
    public string YearsAgo { get; set; }
    /// 
Gets or sets the term for "ages ago".
 
    public string AgesAgo { get; set; }

    /// 
/// Gets or sets the threshold beyond which the fuzzy date should be /// considered "ages ago". ///
 
    public TimeSpan AgesAgoThreshold { get; set; }

    /// 
/// Initialises a new instance with the /// specified properties. ///
 
    private void Initialise(string justNow, string minutesAgo,
        string oneHourAgo, string hoursAgo, string yesterday, string daysAgo,
        string lastMonth, string monthsAgo, string lastYear, string yearsAgo,
        string agesAgo, TimeSpan agesAgoThreshold)
    { ... }
}

FuzzyDateString कक्षा में निम्न शामिल हैं:

public static class FuzzyDateExtensions
{
    public static string ToFuzzyDateString(this TimeSpan timespan)
    {
        return timespan.ToFuzzyDateString(new Grammar());
    }

    public static string ToFuzzyDateString(this TimeSpan timespan,
        Grammar grammar)
    {
        return GetFuzzyDateString(timespan, grammar);
    }

    public static string ToFuzzyDateString(this DateTime datetime)
    {
        return (DateTime.Now - datetime).ToFuzzyDateString();
    }

    public static string ToFuzzyDateString(this DateTime datetime,
       Grammar grammar)
    {
        return (DateTime.Now - datetime).ToFuzzyDateString(grammar);
    }


    private static string GetFuzzyDateString(TimeSpan timespan,
       Grammar grammar)
    {
        timespan = timespan.Duration();

        if (timespan >= grammar.AgesAgoThreshold)
        {
            return grammar.AgesAgo;
        }

        if (timespan < new TimeSpan(0, 2, 0))    // 2 minutes
        {
            return grammar.JustNow;
        }

        if (timespan < new TimeSpan(1, 0, 0))    // 1 hour
        {
            return String.Format(grammar.MinutesAgo, timespan.Minutes);
        }

        if (timespan < new TimeSpan(1, 55, 0))    // 1 hour 55 minutes
        {
            return grammar.OneHourAgo;
        }

        if (timespan < new TimeSpan(12, 0, 0)    // 12 hours
            && (DateTime.Now - timespan).IsToday())
        {
            return String.Format(grammar.HoursAgo, timespan.RoundedHours());
        }

        if ((DateTime.Now.AddDays(1) - timespan).IsToday())
        {
            return grammar.Yesterday;
        }

        if (timespan < new TimeSpan(32, 0, 0, 0)    // 32 days
            && (DateTime.Now - timespan).IsThisMonth())
        {
            return String.Format(grammar.DaysAgo, timespan.RoundedDays());
        }

        if ((DateTime.Now.AddMonths(1) - timespan).IsThisMonth())
        {
            return grammar.LastMonth;
        }

        if (timespan < new TimeSpan(365, 0, 0, 0, 0)    // 365 days
            && (DateTime.Now - timespan).IsThisYear())
        {
            return String.Format(grammar.MonthsAgo, timespan.RoundedMonths());
        }

        if ((DateTime.Now - timespan).AddYears(1).IsThisYear())
        {
            return grammar.LastYear;
        }

        return String.Format(grammar.YearsAgo, timespan.RoundedYears());
    }
}

उन महत्वपूर्ण चीजों में से एक जो मैं प्राप्त करना चाहता था, साथ ही स्थानीयकरण, यह था कि "आज" का अर्थ केवल "यह कैलेंडर दिन" होगा, इसलिए IsToday , IsThisMonth , < कोड> IsThisYear विधियां इस तरह दिखती हैं:

public static bool IsToday(this DateTime date)
{
    return date.DayOfYear == DateTime.Now.DayOfYear && date.IsThisYear();
}

और गोल करने के तरीके इस तरह हैं (मैंने RoundedMonths शामिल किया है, क्योंकि यह थोड़ा अलग है):

public static int RoundedDays(this TimeSpan timespan)
{
    return (timespan.Hours > 12) ? timespan.Days + 1 : timespan.Days;
}

public static int RoundedMonths(this TimeSpan timespan)
{
    DateTime then = DateTime.Now - timespan;

    // Number of partial months elapsed since 1 Jan, AD 1 (DateTime.MinValue)
    int nowMonthYears = DateTime.Now.Year * 12 + DateTime.Now.Month;
    int thenMonthYears = then.Year * 12 + then.Month;                    

    return nowMonthYears - thenMonthYears;
}

मुझे आशा है कि लोगों को यह उपयोगी और / या दिलचस्प लगेगा: ओ)

0
जोड़ा

क्लाइंट-साइड gwt उपयोग के लिए जावा:

import java.util.Date;

public class RelativeDateFormat {

 private static final long ONE_MINUTE = 60000L;
 private static final long ONE_HOUR = 3600000L;
 private static final long ONE_DAY = 86400000L;
 private static final long ONE_WEEK = 604800000L;

 public static String format(Date date) {

  long delta = new Date().getTime() - date.getTime();
  if (delta < 1L * ONE_MINUTE) {
   return toSeconds(delta) == 1 ? "one second ago" : toSeconds(delta)
     + " seconds ago";
  }
  if (delta < 2L * ONE_MINUTE) {
   return "one minute ago";
  }
  if (delta < 45L * ONE_MINUTE) {
   return toMinutes(delta) + " minutes ago";
  }
  if (delta < 90L * ONE_MINUTE) {
   return "one hour ago";
  }
  if (delta < 24L * ONE_HOUR) {
   return toHours(delta) + " hours ago";
  }
  if (delta < 48L * ONE_HOUR) {
   return "yesterday";
  }
  if (delta < 30L * ONE_DAY) {
   return toDays(delta) + " days ago";
  }
  if (delta < 12L * 4L * ONE_WEEK) {
   long months = toMonths(delta);
   return months <= 1 ? "one month ago" : months + " months ago";
  } else {
   long years = toYears(delta);
   return years <= 1 ? "one year ago" : years + " years ago";
  }
 }

 private static long toSeconds(long date) {
  return date / 1000L;
 }

 private static long toMinutes(long date) {
  return toSeconds(date) / 60L;
 }

 private static long toHours(long date) {
  return toMinutes(date) / 60L;
 }

 private static long toDays(long date) {
  return toHours(date) / 24L;
 }

 private static long toMonths(long date) {
  return toDays(date) / 30L;
 }

 private static long toYears(long date) {
  return toMonths(date) / 365L;
 }

}
0
जोड़ा
प्रश्न सी # टैग किया गया है। यह जावा कोड क्यों? आईएमएचओ, केवल सी # कोड लागू होता है
जोड़ा लेखक Kiquenet, स्रोत

Nuget पर Humizer नामक एक पैकेज भी है और यह वास्तव में वास्तव में अच्छी तरह से काम करता है

DateTime.UtcNow.AddHours(-30).Humanize() => "yesterday"
DateTime.UtcNow.AddHours(-2).Humanize() => "2 hours ago"

DateTime.UtcNow.AddHours(30).Humanize() => "tomorrow"
DateTime.UtcNow.AddHours(2).Humanize() => "2 hours from now"

TimeSpan.FromMilliseconds(1299630020).Humanize() => "2 weeks"
TimeSpan.FromMilliseconds(1299630020).Humanize(3) => "2 weeks, 1 day, 1 hour"

Scott Hanselman has a writeup on it on his blog

0
जोड़ा
बहुत उपयोगी! यह उत्तर इस सूची में बहुत अधिक होना चाहिए। अगर मेरे पास 100 वोट थे, तो मैं इसे देता हूं। जाहिर है (जेएस-भूमि से आ रहा है), इस पैकेज की खोज करना आसान नहीं था।
जोड़ा लेखक kumar_harsh, स्रोत
दोस्ताना नोट: .NET 4.5 या ऊपर पूर्ण मानवमार्ग स्थापित नहीं करें ... केवल मानवता स्थापित करें। इसका अधिक हिस्सा .. कारण अन्य भाषा संकुल इस संस्करण पर समर्थित नहीं हैं
जोड़ा लेखक Ahmad, स्रोत

जब आप दर्शक के समय क्षेत्र को जानते हैं, तो दिन के पैमाने पर कैलेंडर दिनों का उपयोग करना स्पष्ट हो सकता है। मैं .NET पुस्तकालयों से परिचित नहीं हूं इसलिए मुझे नहीं पता कि आप सी # में दुर्भाग्य से ऐसा कैसे करेंगे।

उपभोक्ता साइटों पर, आप एक मिनट के भीतर हाथ से चलने वाले भी हो सकते हैं। "एक मिनट पहले से कम" या "अभी अभी" काफी अच्छा हो सकता है।

0
जोड़ा

jquery.timeago प्लगइन

Jeff, because Stack Overflow uses jQuery extensively, I recommend the jquery.timeago प्लगइन .

लाभ:

  • 10 मिनट पहले पृष्ठ खोला गया था, भले ही "1 मिनट पहले" टाइमस्टैम्प से बचें; टाइमगो स्वचालित रूप से रीफ्रेश हो जाता है।
  • आप अपने वेब अनुप्रयोगों में पृष्ठ और / या खंड कैशिंग का पूर्ण लाभ ले सकते हैं, क्योंकि टाइमस्टैम्प की गणना सर्वर पर नहीं की जाती है।
  • आप अच्छे बच्चों की तरह microformats का उपयोग करने के लिए मिलता है।

बस इसे अपने टाइमस्टैम्प पर डीओएम तैयार करें:

jQuery(document).ready(function() {
    jQuery('abbr.timeago').timeago();
});

यह सभी abbr तत्वों को टाइमगो के वर्ग के साथ बदल देगा और एक आईएसओ 8601 शीर्षक में टाइमस्टैम्प:

July 17, 2008

इस तरह कुछ में:

4 months ago

जो पैदा करता है: 4 महीने पहले। जैसे ही समय बीतता है, टाइमस्टैम्प स्वचालित रूप से अपडेट हो जाएंगे।

अस्वीकरण: मैंने इस प्लगइन को लिखा है, इसलिए मैं पक्षपातपूर्ण हूं।

0
जोड़ा
@ रोब फोन्सेका-एन्सर - अब यह मुझे भी रो रहा है। सटीक जानकारी दिखाने के लिए प्रति मिनट एक बार अपडेट कैसे होता है, किसी भी तरह से दूसरे बार एक बार टेक्स्ट झपकी से संबंधित?
जोड़ा लेखक Daniel Earwicker, स्रोत
एक महान प्लगइन - बस मज़ेदार के लिए मैंने कार्यक्षमता को कम करके "इसे नीचे slimmed", autoupdating की तरह, और इसे refactormycode उत्तर </ ए> - jsfiddle.net/drzaus/eMUzF
जोड़ा लेखक drzaus, स्रोत
रयान, मैंने सुझाव दिया कि थोड़ी देर पहले टाइमगो का उपयोग करें। जेफ की प्रतिक्रिया ने मुझे रोया, मेरा सुझाव है कि आप बैठ जाएं: stackoverflow.uservoice.com/pages/1722-general/suggestions/…
जोड़ा लेखक Rob Fonseca-Ensor, स्रोत
@RyanMcGeary: यह संभवतः पूछने का गलत स्थान है लेकिन मुझे टाइमएगो का उपयोग करने के बारे में एक प्रश्न है। मैं यूके में हूं (जीएमटी जिसमें डीएसटी शामिल है) और मेरे डेटाबेस में संग्रहीत तिथियां यूटीसी हैं। अगर मैं गर्मियों में यूटीसी के रूप में अपने डीबी में एक तारीख डालता हूं, तो जाहिर है कि समय 1 घंटे तक बाहर हो जाएगा। तो टाइमएगो प्रिंटिंग के बजाय "लगभग 1 मिनट पहले" यह "लगभग 1 घंटा पहले" कहेंगे। क्या टाइमएगो प्लगइन डीएसटी मतभेदों से निपटता है?
जोड़ा लेखक TheCarver, स्रोत
प्रश्न सी # के बारे में है, मैं यह देखने में असफल रहा कि एक jQuery प्लगइन कैसे प्रासंगिक है।
जोड़ा लेखक BartoszKP, स्रोत
और क्या होगा यदि उपयोगकर्ता जावास्क्रिप्ट अक्षम है? jQuery (और जावास्क्रिट आम ​​तौर पर बोल रहा है) केवल क्लाइंट-साइड अनुभव को बेहतर बनाने के लिए उपयोग किया जाना चाहिए, लेकिन आपकी एप्लिकेशन सुविधाओं को क्लाइंट टेक्नोलॉजीज पर निर्भर नहीं होना चाहिए। आप जावास्क्रिप्ट का उपयोग किए बिना इसे पूरा कर सकते हैं।
जोड़ा लेखक Seb, स्रोत
ऑटो-अपडेटिंग को अक्षम करने का विकल्प इतना कठिन नहीं होना चाहिए, है ना? यदि यह पहले से मौजूद नहीं है, तो यह है ... बस एक बार पगेलोड पर टाइमस्टैम्प अपडेट करें और इसके साथ किया जाए।
जोड़ा लेखक Dean Harding, स्रोत
क्या यह समर्थन करता है। सी # डेटाटाइम.यूटीसी अब ?? यह अच्छा होगा
जोड़ा लेखक blackholeearth0_gmail, स्रोत
@PaparazzoKid हां, टाइमगो समय क्षेत्र को ठीक से संभालता है जब तक कि आपका टाइमस्टैम्प उचित ISO8601 टाइमस्टैम्प न हो। लेकिन तुम सही हो यह सवाल पूछने के लिए यह गलत जगह है। इसके बजाय यहां जाएं: github.com/rmm5t/jquery-timeago/issues
जोड़ा लेखक Ryan McGeary, स्रोत
सेब, यदि आपके पास जावास्क्रिप्ट अक्षम है, तो मूल रूप से abbr टैग के बीच डाली गई स्ट्रिंग प्रदर्शित होती है। आम तौर पर, यह आपके इच्छित प्रारूप में सिर्फ एक तिथि या समय है। Timeago कृपा से गिरावट। यह बहुत आसान नहीं मिलता है।
जोड़ा लेखक Ryan McGeary, स्रोत
हे, धन्यवाद रॉब। वह ठीक है। यह मुश्किल से ध्यान देने योग्य है, खासकर जब संक्रमण के दौरान केवल एक संख्या में परिवर्तन होता है, हालांकि SO पृष्ठों में बहुत से टाइमस्टैम्प होते हैं। मैंने सोचा होगा कि कम से कम पेज कैशिंग के लाभों की सराहना की होगी, भले ही वह ऑटो-अपडेट से बचने का विकल्प चुनता हो। मुझे यकीन है कि जेफ प्लगइन को बेहतर बनाने के लिए प्रतिक्रिया भी दे सकता था। मैं arstechnica.com जैसे सॉलिस जानना साइट्स का उपयोग करता हूं।
जोड़ा लेखक Ryan McGeary, स्रोत

यहां एक कार्यान्वयन है जिसे मैंने डेटटाइम क्लास के विस्तार विधि के रूप में जोड़ा है जो भविष्य और पिछली तिथियों दोनों को संभालता है और एक अनुमानित विकल्प प्रदान करता है जो आपको उस स्तर का विवरण निर्दिष्ट करने की अनुमति देता है जिसे आप ढूंढ रहे हैं ("3 घंटे पहले" बनाम "3 घंटे, 23 मिनट, 12 सेकंड पहले "):

using System.Text;

/// 
/// Compares a supplied date to the current date and generates a friendly English /// comparison ("5 days ago", "5 days from now") ///
 
/// 
The date to convert
/// 
When off, calculate timespan down to the second.
/// When on, approximate to the largest round unit of time.
/// 
public static string ToRelativeDateString(this DateTime value, bool approximate)
{
    StringBuilder sb = new StringBuilder();

    string suffix = (value > DateTime.Now) ? " from now" : " ago";

    TimeSpan timeSpan = new TimeSpan(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();
}
0
जोड़ा

आईफोन ओबीजे-सी संस्करण

+ (NSString *)timeAgoString:(NSDate *)date {
int delta = -(int)[date timeIntervalSinceNow];

if (delta < 60)
{
    return delta == 1 ? @"one second ago" : [NSString stringWithFormat:@"%i seconds ago", delta];
}
if (delta < 120)
{
    return @"a minute ago";
}
if (delta < 2700)
{
    return [NSString stringWithFormat:@"%i minutes ago", delta/60];
}
if (delta < 5400)
{
    return @"an hour ago";
}
if (delta < 24 * 3600)
{
    return [NSString stringWithFormat:@"%i hours ago", delta/3600];
}
if (delta < 48 * 3600)
{
    return @"yesterday";
}
if (delta < 30 * 24 * 3600)
{
    return [NSString stringWithFormat:@"%i days ago", delta/(24*3600)];
}
if (delta < 12 * 30 * 24 * 3600)
{
    int months = delta/(30*24*3600);
    return months <= 1 ? @"one month ago" : [NSString stringWithFormat:@"%i months ago", months];
}
else
{
    int years = delta/(12*30*24*3600);
    return years <= 1 ? @"one year ago" : [NSString stringWithFormat:@"%i years ago", years];
}

}

0
जोड़ा
var ts = new TimeSpan(DateTime.Now.Ticks - dt.Ticks);
0
जोड़ा

आप इसे आजमा सकते हैं। मुझे लगता है कि यह सही तरीके से काम करेगा।

long delta = new Date().getTime() - date.getTime();
const int SECOND = 1;
const int MINUTE = 60 * SECOND;
const int HOUR = 60 * MINUTE;
const int DAY = 24 * HOUR;
const int MONTH = 30 * DAY;

if (delta < 0L)
{
  return "not yet";
}
if (delta < 1L * MINUTE)
{
  return ts.Seconds == 1 ? "one second ago" : ts.Seconds + " seconds ago";
}
if (delta < 2L * MINUTE)
{
  return "a minute ago";
}
if (delta < 45L * MINUTE)
{
  return ts.Minutes + " minutes ago";
}
if (delta < 90L * MINUTE)
{
  return "an hour ago";
}
if (delta < 24L * HOUR)
{
  return ts.Hours + " hours ago";
}
if (delta < 48L * HOUR)
{
  return "yesterday";
}
if (delta < 30L * DAY)
{
  return ts.Days + " days ago";
}
if (delta < 12L * MONTH)
{
  int months = Convert.ToInt32(Math.Floor((double)ts.Days / 30));
  return months <= 1 ? "one month ago" : months + " months ago";
}
else
{
  int years = Convert.ToInt32(Math.Floor((double)ts.Days / 365));
  return years <= 1 ? "one year ago" : years + " years ago";
}
0
जोड़ा
/** 
 * {@code date1} has to be earlier than {@code date2}.
 */
public static String relativize(Date date1, Date date2) {
    assert date2.getTime() >= date1.getTime();

    long duration = date2.getTime() - date1.getTime();
    long converted;

    if ((converted = TimeUnit.MILLISECONDS.toDays(duration)) > 0) {
        return String.format("%d %s ago", converted, converted == 1 ? "day" : "days");
    } else if ((converted = TimeUnit.MILLISECONDS.toHours(duration)) > 0) {
        return String.format("%d %s ago", converted, converted == 1 ? "hour" : "hours");
    } else if ((converted = TimeUnit.MILLISECONDS.toMinutes(duration)) > 0) {
        return String.format("%d %s ago", converted, converted == 1 ? "minute" : "minutes");
    } else if ((converted = TimeUnit.MILLISECONDS.toSeconds(duration)) > 0) {
        return String.format("%d %s ago", converted, converted == 1 ? "second" : "seconds");
    } else {
        return "just now";
    }
}
0
जोड़ा

यदि आप "2 दिन, 4 घंटे और 12 मिनट पहले" जैसे आउटपुट चाहते हैं, तो आपको एक समय की आवश्यकता है:

TimeSpan timeDiff = DateTime.Now-CreatedDate;

फिर आप अपनी पसंद के मूल्यों तक पहुंच सकते हैं:

timeDiff.Days
timeDiff.Hours

आदि।

0
जोड़ा

यह मेरा काम है, एक आकर्षण की तरह काम करता है :)

public static string RelativeDate(DateTime theDate)
        {
            var span = DateTime.Now - theDate;
            if (span.Days > 365)
            {
                var years = (span.Days / 365);
                if (span.Days % 365 != 0)
                    years += 1;
                return $"about {years} {(years == 1 ? "year" : "years")} ago";
            }
            if (span.Days > 30)
            {
                var months = (span.Days / 30);
                if (span.Days % 31 != 0)
                    months += 1;
                return $"about {months} {(months == 1 ? "month" : "months")} ago";
            }
            if (span.Days > 0)
                return $"about {span.Days} {(span.Days == 1 ? "day" : "days")} ago";
            if (span.Hours > 0)
                return $"about {span.Hours} {(span.Hours == 1 ? "hour" : "hours")} ago";
            if (span.Minutes > 0)
                return $"about {span.Minutes} {(span.Minutes == 1 ? "minute" : "minutes")} ago";
            if (span.Seconds > 5)
                return $"about {span.Seconds} seconds ago";

            return span.Seconds <= 5 ? "about 5 seconds ago" : string.Empty;
        }
0
जोड़ा
खुश कोडिंग: डी
जोड़ा लेखक VnDevil, स्रोत
एक जादू की तरह काम करता है :)
जोड़ा लेखक densityx, स्रोत