1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950 |
- /**
- * https://ru.wikibooks.org/wiki/%D0%A0%D0%B5%D0%B0%D0%BB%D0%B8%D0%B7%D0%B0%D1%86%D0%B8%D0%B8_%D0%B0%D0%BB%D0%B3%D0%BE%D1%80%D0%B8%D1%82%D0%BC%D0%BE%D0%B2/%D0%92%D0%B5%D1%87%D0%BD%D1%8B%D0%B9_%D0%BA%D0%B0%D0%BB%D0%B5%D0%BD%D0%B4%D0%B0%D1%80%D1%8C#C,_C++
- * получения дня недели из даты
- * 0 - вск, 1-6 - пнд-сбт.
- */
- typedef unsigned short Year;
- typedef unsigned char Month;
- typedef unsigned char Day;
- typedef unsigned char Weekday;
- const char* weekdays[] = {"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"};
- Weekday weekday(Year year, Month month, Day day) {
- if (month < 3u) {
- --year;
- month += 10u;
- } else
- month -= 2u;
- return (Weekday)((day + 31u * month / 12u + year + year / 4u - year / 100u + year / 400u) % 7u);
- }
- /**
- * https://blog.reverberate.org/2020/05/12/optimizing-date-algorithms.html
- * Optimizing UTC → Unix Time Conversion For Size And Speed
- */
- /* https://github.com/protocolbuffers/upb/blob/22182e6e/upb/json/parser.rl#L1697
- * epoch_days(1970, 1, 1) == 1970-01-01 == 0 */
- int epoch_days_table(int year, int month, int day) {
- static const uint16_t month_yday[12] = {0, 31, 59, 90, 120, 151,
- 181, 212, 243, 273, 304, 334};
- uint32_t year_adj = year + 4800; /* Ensure positive year, multiple of 400. */
- uint32_t febs = year_adj - (month <= 2 ? 1 : 0); /* Februaries since base. */
- uint32_t leap_days = 1 + (febs / 4) - (febs / 100) + (febs / 400);
- uint32_t days = 365 * year_adj + leap_days + month_yday[month - 1] + day - 1;
- return days - 2472692; /* Adjust to Unix epoch. */
- }
- /* https://github.com/protocolbuffers/upb/blob/22182e6e/upb/json_decode.c#L982
- * epoch_days_fast(1970, 1, 1) == 1970-01-01 == 0. */
- int epoch_days_fast(int y, int m, int d) {
- const uint32_t year_base = 4800; /* Before min year, multiple of 400. */
- const uint32_t m_adj = m - 3; /* March-based month. */
- const uint32_t carry = m_adj > m ? 1 : 0;
- const uint32_t adjust = carry ? 12 : 0;
- const uint32_t y_adj = y + year_base - carry;
- const uint32_t month_days = ((m_adj + adjust) * 62719 + 769) / 2048;
- const uint32_t leap_days = y_adj / 4 - y_adj / 100 + y_adj / 400;
- return y_adj * 365 + leap_days + month_days + (d - 1) - 2472632;
- }
|