如何从C中获取一年中的日期和iso周#

时间:2018-01-12 22:02:44

标签: c date

我在网上找到的所有这些例子似乎都使用的语言在C语言中没有明显的类比。我需要一个可以在普通C语言中使用的解决方案,而不依赖于标准语言之外的任何外部库。

考虑到年份数和iso周数,我想知道如何计算这些值的实际日期(日,月和年)。

编辑: 谢谢你到目前为止的答案,但我认为可能不清楚我最初想要问的是什么。我不是在谈论自特定年份的1月1日以来的几个星期,我特别谈论iso周数。例如,2019年第1周的第1周开始于2018年12月31日......等周总是从星期一开始,其开始日期的年份数有时可以从实际日历年中减去1。

1 个答案:

答案 0 :(得分:2)

  

如何从一年中获得一个日期,如何在C

中获得iso周#

秘密在于ISO年的1月4日Jan 4 is always in week 1

查找1月4日的星期几。然后查找自上周一以来的日期 -  那天开始ISO年。每周加7(偏移1)。

#include <stdio.h>
#include <stdlib.h>
#include <time.h>

typedef struct {
  int year, month, day;
} ymd;

typedef struct {
  int year, week, dow; // ISO week-date: year, week, day-of-the-week
} ISO_week_date;

int ISO_week_date_to_ymd(ymd *y, const ISO_week_date *x) {
  // Set to noon, Jan 4 of the year.
  struct tm tm = {.tm_year = x->year - 1900, .tm_mon = 0, .tm_mday = 4,
      .tm_hour = 12};
  // Use mktime() to find the day-of-the week
  if (mktime(&tm) == -1) {
    return -1;
  }
  // Sunday to Jan 4
  int DaysSinceSunday = tm.tm_wday;
  // Monday to Jan 4
  int DaysSinceMonday = (DaysSinceSunday + (7 - 1)) % 7;
  tm.tm_mday += x->dow + (x->week - 1) * 7 - DaysSinceMonday;
  if (mktime(&tm) == -1) {
    return -1;
  }
  y->year = tm.tm_year + 1900;
  y->month = tm.tm_mon + 1;
  y->day = tm.tm_mday;
  return 0;
}

测试

char *dow_name(ymd date) {
  struct tm tm = {.tm_year = date.year - 1900, .tm_mon = date.month - 1,
      .tm_mday = date.day, .tm_hour = 12};
  mktime(&tm);
  static char name[4];
  sprintf(name, "%.3s", asctime(&tm));
  return name;
}

void test(int year, int week, int dow) {
  ISO_week_date week_date = {.year = year, .week = week, .dow = dow};
  ymd date;
  if (ISO_week_date_to_ymd(&date, &week_date)) {
    fprintf(stderr, "Oops\n");
    exit(EXIT_FAILURE);
  }
  printf("%04d-W%02d-%d --> %s %04d-%02d-%02d\n", year, week, dow,
      dow_name(date), date.year, date.month, date.day);
}

int main(void) {
  for (int year = 2012; year <= 2018; year++) {
    test(year, 1, 0);
    test(year, 52, 6); // Some ISO week-numbering years have 53 weeks 
    puts("");
  }
}

输出

2012-W01-0 --> Mon 2012-01-02
2012-W52-6 --> Sun 2012-12-30

2013-W01-0 --> Mon 2012-12-31
2013-W52-6 --> Sun 2013-12-29

2014-W01-0 --> Mon 2013-12-30
2014-W52-6 --> Sun 2014-12-28

2015-W01-0 --> Mon 2014-12-29
2015-W52-6 --> Sun 2015-12-27

2016-W01-0 --> Mon 2016-01-04
2016-W52-6 --> Sun 2017-01-01

2017-W01-0 --> Mon 2017-01-02
2017-W52-6 --> Sun 2017-12-31

2018-W01-0 --> Mon 2018-01-01
2018-W52-6 --> Sun 2018-12-30

注意:

代码使用午间来应对夏令时可能导致的调整。我特别担心tm.tm_mday += ....可能会进行日光调整。或者,设置.tm_isdst = 0可能就足够了。