Funktion zur Berechnung des Wochentages


  • Mod

    Wenn man unsere Formel optisch etwas pusht, die korrekte isLeapyear(...) einsetzt und die korrekte Zahl der Tage pro Jahr verwendet, dann hat das auch etwas Eingängliches.

    uint8_t calculateWeekday(uint16_t year, uint8_t month, int32_t day)
    {
        day += 6; // 1.1.2000 was a saturday
        day += (year-2000) * 365.24219;
    
        if      (month > 11)    day+=334;
        else if (month > 10)    day+=304;
        else if (month >  9)    day+=273;
        else if (month >  8)    day+=243;
        else if (month >  7)    day+=212;
        else if (month >  6)    day+=181;
        else if (month >  5)    day+=151;
        else if (month >  4)    day+=120;
        else if (month >  3)    day+=90;
        else if (month >  2)    day+=59;
        else if (month >  1)    day+=31;
    
        if ((year % 4 == 0) && ((year % 100 != 0) || (year % 400 == 0)) && (month < 2 || (month == 2 && day <= 28)))
        {
            day--;
        }
    
        return ( day % 7 + 1 );
    }
    

    @dot: Diese perfekte Formel von Gauß versteht man nicht leicht. Es gibt übrigens noch mehr Systeme zur Berechnung.

    @neuer_user: Vielleicht nehmen wir auch deine Formel. Noch ist alles offen.



  • Erhard Henkes schrieb:

    Diese perfekte Formel versteht man nicht. Es gibt übrigens noch mehr Systeme zur Berechnung.

    Was genau versteht man daran nicht, im Wikipedia-Artikel ist doch eine komplette Herleitung!?


  • Mod

    Welche Formel wir verwenden werden, entscheidet letztendlich das Team. Der Thread ist noch zu frisch. Danke auf jeden Fall für den Tipp. 🙂



  • Meine Version ist länger.
    Mich stört diese Kommaangabe

    (year-2000) * [u]365.24219[/u];
    

  • Mod

    Ja, das stimmt. Man könnte vielleicht auch 146097.0 / 400.0 verwenden.

    Wenn wir eine kürzere Formel für month ==> day finden, ist unsere Formel viel schöner. Sie muss ja nur ab 2000 und nicht vorher gelten.

    - isLeapyear(...) könnte man auslagern
    - die Zahlen der Tage für einen abgeschlossenen Monat könnte man in ein Array packen



  • Erhard Henkes schrieb:

    Sie muss ja nur ab 2000 und nicht vorher gelten.

    So haben schon viele gedacht... 😉


  • Mod

    static uint16_t days[12] = {  0,  31,  59, 90, 120, 151, 181, 212, 243, 273, 304, 334};
    
    static bool isLeapyear(uint16_t year)
    {
        return (year % 4 == 0) && ((year % 100 != 0) || (year % 400 == 0));
    }
    
    static uint8_t calculateWeekday(uint16_t year, uint8_t month, int32_t day)
    {
        day += 6; // 1.1.2000 was a saturday
        day += (year-2000) * 365.24219 + days[month-1];
    
        if (isLeapyear(year) && (month < 2 || (month == 2 && day <= 28)))
        {
            day--;
        }
    
        return ( day % 7 + 1 );
    }
    

    So sieht das doch schon richtig fein aus. 😃

    Verflixt! beim 31.12.2099 kommt Mittwoch raus, ist aber Donnerstag. Die genaue Kommazahl ist offenbar nicht hilfreich. Nein, daran liegt es nicht.



  • Erhard Henkes schrieb:

    Verflixt! beim 31.12.2099 kommt Mittwoch raus, ist aber Donnerstag. Die genaue Kommazahl ist offenbar nicht hilfreich. Nein, daran liegt es nicht.

    Woher kommt die 365.24219?


  • Mod



  • Sag mir nicht, daß Du da einfach das tropische Jahr aus dem Lexikon kopiert hast. Das darf ja nicht wahr sein.

    Jedes Jahr hat 365 Tage. 365
    Jedes vierte hat einen Tag mehr. +0.25
    Jedes hundertste einen Tag weniger. -0.01
    Jedes vierhundertste einen mehr. +0.0025
    sind für mich 365.2425

    Erhard schrieb:

    146097.0 / 400.0

    Ach nööö!


  • Mod

    Ich denke, es liegt daran, dass ich bei dem falsch eingestellten Datum kompiliert habe. Nach rebuild in 2011 klappt es wieder.

    Die Zahl dort ist nicht entscheidend, ob 365.2419, 365.2425 oder 365.25 😃

    Wir nehmen aber Volkard's Zahl, weil die nachvollziehbar ist (wenn sie auch nicht stimmt):

    static uint16_t days[12] = {  0,  31,  59, 90, 120, 151, 181, 212, 243, 273, 304, 334};
    
    static bool isLeapyear(uint16_t year)
    {
        return (year % 4 == 0) && ((year % 100 != 0) || (year % 400 == 0));
    }
    
    static uint8_t calculateWeekday(uint16_t year, uint8_t month, int32_t day)
    {
        day += 6; // 1.1.2000 was a saturday
        day += (year-2000) * 365.2425 + days[month-1];
    
        if (isLeapyear(year) && (month < 2 || (month == 2 && day <= 28)))
        {
            day--;
        }
    
        return ( day % 7 + 1 );
    }
    

    31.12.2099 ist ein Donnerstag! yep.



  • Ich weiß schon, weshalb ich bei Dir nicht mitspiele.


  • Mod

    Ich weiß schon, weshalb ich bei Dir nicht mitspiele.

    Eigentlich schade. Mit dir als Math Geek könnte das richtig gut werden, aber wir schaffen das auch so. Ist sowieso mehr Praxis als Theorie. 😉

    Diese Wochentag-Formel ist ursprünglich nicht von mir, ich wollte sie nur verfeinern oder zumindest korrekt machen.

    Entscheidend ist, dass der korrekte Tag heraus kommt, und das klappt nun.



  • Erhard Henkes schrieb:

    Wir nehmen aber Volkard's Zahl, weil die nachvollziehbar ist (wenn sie auch nicht stimmt)

    Sie stimmt natürlich, weil der gregorianische Kalender abgebildet werden soll und nichts anderes.

    Naja, Du hattest die richtige Zahl schon in der Hand: 146097.0/400.0=365.2425


  • Mod

    OK, dann nehmen wir den Bruch. Dann ist neuer_user auch zufrieden. 🙂

    static uint8_t calculateWeekday(uint16_t year, uint8_t month, int32_t day)
    {
        day += 6; // 1.1.2000 was a saturday
        day += (year-2000) * 146097.0/400.0 + days[month-1];
    
        if (isLeapyear(year) && (month < 2 || (month == 2 && day <= 28)))
        {
            day--;
        }
    
        return ( day % 7 + 1 );
    }
    

  • Mod

    Ich finde, dass sich diese Formel nun sehen lassen kann. Wenn man im Datum zurück geht, kann man diese nun sicher auch ab 1980 einsetzen.



  • Da sich der Kalender alle 400 Jahre wiederholt, kannste statt j-2000 auch j-1600 schreiben. Oder j-0.


  • Mod

    Danke für den Hinweis. Mit meinem Win XP SP3 kann ich nur von 1.1.1980 bis 31.12.2099 experimentell mit VBox testen. Qemu macht die Grätsche (schwach gebastelt), und GNU GCC versagt beim Compilieren, wenn die Zeit aus den Fugen ist. 😃

    Habe gerade bei meinem Sohn Win 7 gecheckt, selber Murks: 1980 - 2099. 😃


  • Mod

    http://www.thkoehler.de/midnightblue/m_kal.htm

    Volkard hat Recht. 1.1.1600 ist ein Samstag. 🙂

    @Volkard: 0 ist allerdings nicht brauchbar wegen dem 15.10.1582 (Beginn der gregorianischen Zeitrechnung). Da hat der gute Gregor 10 Tage ausgelassen (4.10. ==> 15.10.1582 ).



  • Erhard Henkes schrieb:

    @Volkard: 0 ist allerdings nicht brauchbar wegen dem 15.10.1582 (Beginn der gregorianischen Zeitrechnung). Da hat der gute Gregor 10 Tage ausgelassen (4.10. ==> 15.10.1582 ).

    Doch.
    Es geht immernoch nur darum, den gregorianischen Kalender abzubilden. Für Eingaben nach dem 15.10.1582 klappt doch alles mit j-0.


Anmelden zum Antworten