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

Posted

技术标签:

【中文标题】如何在C中从一年和iso周中获取日期#【英文标题】:How to get a date from a year and iso week # in C 【发布时间】:2018-01-12 22:02:43 【问题描述】:

我可以在网上找到的所有示例似乎都使用了具有在 C 中没有明显类比功能的语言。我需要一个可以在纯 C 中工作的解决方案,而不依赖于标准库之外的任何外部库。

给定年份编号和 iso 周编号,我想知道如何计算这些值的实际日期(日、月和年)。

编辑: 到目前为止,感谢您的回答,但我认为可能不清楚我最初想问的是什么。我不是在谈论自给定年份的 1 月 1 日以来的周数,而是专门谈论 iso 周数。例如,2019 年的 iso 第 1 周从 2018 年 12 月 31 日开始...iso 周总是从星期一开始,其开始日期的年份编号有时可能与实际日历年相差 1。

【问题讨论】:

time.h使用localtimelocaltime_rgmtimegmtime_r 更具体地说,您可以通过在 struct tm 中填写您想要的年份(当然是减去 1900)、tm_mon 包含 0 和 @987654332 来获得相当远的距离@ 包含 7 倍的星期数,加上星期几。然后拨打mktime()。完成此操作后,tm_montm_mdaytm_wday 应该包含正确的日期值。 @SteveSummit:所以基本上是反向运行this answer。 How do I calculate the week number given a date? 中提供了有用的信息。我的答案包括代码(它实际上是 SPL——存储过程语言(用于 Informix)——但您可以将其视为伪代码。它并不难理解(尽管编写代码使其工作起来可能很痛苦) )。 有一两天前有人问过一个相关问题 — C week of year with variable start and anchor day。它更多地是关于按周进行而不是相反,而且您可能并不追求它所要求的灵活性。 【参考方案1】:

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

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

注意:周一是 ISO 周一至周日周的第一天。

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

[编辑] 代码在 2020 年 6 月 30 日重新工作,@@JamesK.Lowden 正确指出代码在应该是 1-7 时使用的是星期几 0-6。这并不影响 OP 的年/周数目标。

#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 1-52,53, day-of-the-week 1-7
 ISO_week_date;

int ISO_week_date_to_ymd(ymd *y, const ISO_week_date *x) 
  // Set to noon, Jan 4 of the year to avoid daylight saving time issues
  // Jan 4 is always in the 1st week 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; // .tm_wday is days since Sunday
  // Monday to Jan 4
  int DaysSinceMonday = (DaysSinceSunday + (7 - 1)) % 7;
  tm.tm_mday += (x->dow - 1) + (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, 1);
    test(year, 52, 7); // Some ISO week-numbering years have 53 weeks
    puts("");
  

  for (int dow = 1; dow <= 7; dow++) 
    test(2020, 27, dow);
  

输出

2012-W01-1 --> Mon 2012-01-02
2012-W52-7 --> Sun 2012-12-30

2013-W01-1 --> Mon 2012-12-31
2013-W52-7 --> Sun 2013-12-29

2014-W01-1 --> Mon 2013-12-30
2014-W52-7 --> Sun 2014-12-28

2015-W01-1 --> Mon 2014-12-29
2015-W52-7 --> Sun 2015-12-27

2016-W01-1 --> Mon 2016-01-04
2016-W52-7 --> Sun 2017-01-01

2017-W01-1 --> Mon 2017-01-02
2017-W52-7 --> Sun 2017-12-31

2018-W01-1 --> Mon 2018-01-01
2018-W52-7 --> Sun 2018-12-30

2020-W27-1 --> Mon 2020-06-29
2020-W27-2 --> Tue 2020-06-30
2020-W27-3 --> Wed 2020-07-01
2020-W27-4 --> Thu 2020-07-02
2020-W27-5 --> Fri 2020-07-03
2020-W27-6 --> Sat 2020-07-04
2020-W27-7 --> Sun 2020-07-05

注意事项:

代码使用中午来应对由于夏令时而可能发生的调整。我特别担心tm.tm_mday += .... 可能会越过日光调整。或者设置.tm_isdst = 0 可能就足够了。

【讨论】:

虽然很有帮助,但我不确定代码对于 ISO 格式是否正确。 Wikipedia 示例和tondering.dk/claus/cal/iso8601.php 都表明这种格式的 ISO 日是 1-7,而不是所示的 0-6。今天是 6 月 30 日:2020-W27-2。这也是今天 GNU 日期实用程序所显示的内容。该计划将 2020-W27-2 转换为 2020 年 7 月 1 日。

以上是关于如何在C中从一年和iso周中获取日期#的主要内容,如果未能解决你的问题,请参考以下文章

如何获取特定月份的每个星期四或一周中的其他日期的日期?

PHP 如何获取两个时间之间的年和月份及间隔天数 PHP两个日期之间的所有日期

获取一周中特定日期的下一个日期

如何在django模板中从给定日期起特定天数后获取日期

如何在java中从时区获取日期[重复]

如何在 swift 中从两天中获取日期名称