繁体   English   中英

在C ++中将unix时间戳转换为星期几?

[英]Convert unix timestamp (s) to day of week in C++?

如何根据任意Unix时间戳(秒)确定加利福尼亚州(太平洋时间)的星期几? 我搜索过,但还没有找到C ++的内置库。

UTC通常比PT提前8小时,但只是从Unix时间戳减去8小时并且创建一个tm结构不起作用,因为这可以减少夏令时的细微差别。

以下是使用Howard Hinnant的日期库的方法 这是一个基于<chrono>的免费MIT许可的C ++ <chrono>库:

#include "tz.h"
#include <iostream>

int
main()
{
    using namespace std::chrono_literals;
    using namespace date;
    auto unix_timestamp = sys_seconds{1479664467s};
    auto zt = make_zoned("America/Los_Angeles", unix_timestamp);
    auto wd = weekday{floor<days>(zt.get_local_time())};
    std::cout << wd << '\n';
}

第一行只构造unix时间戳(您可能从其他来源获得)。 在C ++ 11中,我们没有chrono-literals,因此这行代替:

auto unix_timestamp = sys_seconds{seconds{1479664467}};

make_zoned()根据unix_timestamptime_zone “America / Los_Angeles”创建zoned_time<seconds> zoned_timetime_zone和unix时间戳的集合。

您可以使用zt.get_local_time()zoned_time获取本地时间。 这是一个chrono::time_point但没有与之关联的时钟。 此时间点的精度将等于原始时间戳的精度(在这种情况下为秒)。

你可以得到的一周的当地日local_time通过截断local_time到的精确days

floor<days>(zt.get_local_time())

而这一天精确度time_point将转换为weekday类型。

可以打印weekday ,参加weekday算术和比较,或者在[0,6]范围内显式转换为unsigned

上述程序输出:

Sun

此库的时区部分是IANA时区数据库的忠实代表。 它将在当地时区和UTC之间进行正确的调整,直到1800年代中期到现在。 如果您在1800年代中期之前传递库时间戳,则最早的已知UTC偏移量会向后推断到-32768年。 您还可以将未来的时间戳传递给32767年,并且当前的偏移和夏令时规则将向前传播。

由于这个库是基于<chrono>构建的,它将处理<chrono>提供的任何精度。 请注意,某些chrono::duration s(如nanoseconds范围更为有限( nanoseconds为+/- 292年)。

库/进程是线程安全的。 也就是说,程序可以直接使用您指定的任何时区,而无需更改计算机的时区或环境变量。 它也不涉及C时序API的一部分,由于非const函数局部静态,已知它不是线程安全的。

移植到最新版本的gcc,clang和VS.

使用标准UNIX C(非线程安全版)的Hacky方式:

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

void print_times() {
  char my_buf[100];
  time_t now = time(NULL);
  printf(ctime(&now));
  strftime(my_buf, 100, "DoW=%w Tz=%Z\n", localtime(&now));
  printf(my_buf);
}

int main( int argc, char * argv[] ) {
  print_times();
  setenv("TZ", "America/Los_Angeles", 1);
  print_times();
  unsetenv("TZ");
  print_times();
}

不幸的是, TZ env var或更改文件/etc/localtime是影响UNIX时间函数的时区设置的唯一方法(在引擎盖下tzset()使用这些源来初始化一组编码时区设置的共享变量所有的库函数)。

问题是您需要为您需要的区域提供UNIX TZ字符串。 zoneinfo格式是最简单和推荐的格式。 我用tzselect生成太平洋时间的TZ值。

另请参阅man 5 localtimeman tzselect了解有关标准UNIX时区设置的信息。

你可以使用Boost库吗? 如果您可以找出Posix时区字符串(IEEE Std 1003.1,请参阅http://www.boost.org/doc/libs/1_62_0/doc/html/date_time/local_time.html#date_time.local_time.posix_time_zone )时区(PST),然后下面的例子可能有所帮助。

#include <iostream>
#include <ctime>
#include <boost/date_time.hpp>

int main(int argc, char ** argv)
{
    std::string posix_time_zone_string = "EST-05:00:00";  // you need to change this to the posix time representation of your desired timezone

    time_t unix_time = 1479641855;  //1479641855 = Sun, 20 Nov 2016 11:37:35 GMT

    boost::posix_time::ptime pt = boost::posix_time::from_time_t(unix_time);
    std::cout << "time in UTC: " << boost::posix_time::to_iso_extended_string(pt) << std::endl;


    boost::local_time::time_zone_ptr zone(new boost::local_time::posix_time_zone(posix_time_zone_string));

    boost::local_time::local_date_time dt_with_zone(pt, zone);
    std::cout << "time in local timezone: " << boost::posix_time::to_iso_extended_string(dt_with_zone.local_time()) << std::endl;

    // Get the week day (alternative 1)
    std::cout << "local week day integer: " << boost::local_time::to_tm(dt_with_zone).tm_wday << std::endl;   // 0=sunday, 1=monday, etc.

    // Get the week day (alternative 2)
    std::cout << "local week day name: " << dt_with_zone.local_time().date().day_of_week() << std::endl;
    std::cout << "local week day integer: " << int(dt_with_zone.local_time().date().day_of_week()) << std::endl;
    return 0;
}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM