我们的最终目的是把字符串格式的时间转换为内部使用的“日历时间”,即到UTC 1970年1月1日零时的秒数。这里就存在夏令时的问题。比如俄罗斯时间2008年10月26日2:30由于夏令时的跳变会经过两次,这两次所代表的“日历时间”明显不同。如果仍按照上面的程序,由mktime()函数来决定这个时间是否处于夏令时(它会根据当前时区自动判断,在没有发生时间重叠的情况下处理是正确的)就会有问题。这时我们不能使用ttm.tm_isdst = -1了,而是明确使用ttm_tm_isdst = 1来告知mktime()现在处于夏令时,明确使用ttm.tm_isdst = 0来告知未处于夏令时。 
结论:使用字符串时间表示时一定要有标记表示这个时间是否处于夏令时,推荐在时间字符串后面添加DST或者加上时区。否则,无法正确处理夏令时情况。如果时间字符串表示的是UTC时间,则无需上述处理过程。

mktime各种情况测试一下

在支持不支持夏令时时区,夏令时区间,非夏令时区间,切入夏令时丢失小时时,切出夏令时重复小时时。。tm_isdst 的不同作用

mktime是用于生成指定的本地时刻的,多数情况下我们都会把tm_isdst设置为-1,这样子可以把指定时刻是否为夏令时交由系统判断.不过实际应用中发现,对于切出夏令时时出现的重复小时,一个小时处于夏令时,一个小时不处于夏令时,因为我们没有指定是否处于夏令时,这时得到的结果便不稳定,可能是处于夏令时的小时,也可能是处于非夏令时的小时.

下面是测试结果:
    tm中tm_isdst分别为0,1,-1对mktime结果影响.
    tm_isdst = -1(由系统判断)
    对于切入夏令时丢失的小时前的小时,生成时间为指定时间.
    对于切入夏令时丢失的小时,生成时间为指定小时后面一个小时(不受前面mk结果影响)
    对于切入夏令时丢失的小时后的小时,生成时间为指定时间.

对于切出夏令时重复的小时前的小时,生成时间为指定时间.
    对于切出夏令时重复的小时,生成时间受前面mk结果影响,倘若前次mk时间结果处于夏令时,则结果为处于夏令时的那个小时;倘若前次mk时间结果不处于夏令时,则结果为不处于夏令时的那个小时;
    对于切出夏令时重复的小时后的小时,生成时间为指定时间.

tm_isdst = 0(指定为非夏令时)
    对于切入夏令时丢失的小时前的小时,生成时间为指定时间.
    对于切入夏令时丢失的小时,生成时间为指定小时后面一个小时(不受前面mk结果影响)
    对于切入夏令时丢失的小时后的小时,生成时间为指定时间后面一个小时.

对于切出夏令时重复的小时前的小时,生成时间为指定时间后面一个小时.
    对于切出夏令时重复的小时,生成时间为不处于夏令时的那个小时;
    对于切出夏令时重复的小时后的小时,生成时间为指定时间.

tm_isdst = 1(指定为夏令时)
    对于切入夏令时丢失的小时前的小时,生成时间为指定时间前面一个小时.
    对于切入夏令时丢失的小时,生成时间为指定小时后面一个小时(不受前面mk结果影响)
    对于切入夏令时丢失的小时后的小时,生成时间为指定时间.

对于切出夏令时重复的小时前的小时,生成时间为指定时间.
    对于切出夏令时重复的小时,生成时间为处于夏令时的那个小时;
对于切出夏令时重复的小时后的小时,生成时间为指定时间前面一个小时.

倘若mktime传入时间因为夏令时存在两个,那mktime会认为该时间可能处于夏令时也可能处于非夏令时。是否处于夏令时由mktime生成的上一个时间决定,倘若上一个时间处于夏令时则生成的该重复时间是处于夏令时的,否则生成的该重复时间是处于非夏令时的。程序内部的原因是它在mktime中用一个静态变量保存夏令时状态,下次生成时间是优先以保存的状态来生成时间。

倘若mktime一个不存在的夏令时(出现夏令时切换时丢失的那个小时)会直接生成丢失小时的下一个小时。这个结果是确定的稳定的。

暂时解决问题的一个方案:

#include <iostream>
#include <time.h>
#include <arpa/inet.h>
using namespace std;
time_t MakeTime(int iYear, int iMonth, int iMonthDay,
int iHour, int iMinute, int iSecond, int iIsdst)
{
struct tm dTm;
dTm.tm_mon = iMonth - 1;
dTm.tm_mday = iMonthDay ;
dTm.tm_year = iYear - 1900;
dTm.tm_hour = iHour;
dTm.tm_min = iMinute;
dTm.tm_sec = iSecond;
dTm.tm_isdst = iIsdst; time_t ret = mktime(&dTm); // 判断生成的是不是夏令时中的,倘若系统一直不是夏令时,可以省一些性能
if(1 == dTm.tm_isdst)
{
// 调整为非夏令时
dTm.tm_isdst = 0;
time_t tmpRet = mktime(&dTm);
if(iHour == dTm.tm_hour)
{
// 确定为重复时间
ret = tmpRet;
}
}
return ret; } string ToString(time_t tTime)
{
struct tm Tm;
struct tm* pTm;
pTm = localtime_r(&tTime, &Tm);
if(NULL == pTm)
{
}
int iSecond = pTm->tm_sec;
int iMinute = pTm->tm_min;
int iHour = pTm->tm_hour;
int iMonthDay = pTm->tm_mday;
int iMonth = pTm->tm_mon + 1;
int iYear = pTm->tm_year + 1900;
#define MAX_DATETIME_STRING_FORMAT "%04d-%02d-%02d %02d:%02d:%02d %s"
static const int MaxDateTimeStringLength = 23;
char temp[MaxDateTimeStringLength + 1];
snprintf(temp, MaxDateTimeStringLength + 1, MAX_DATETIME_STRING_FORMAT, iYear, iMonth, iMonthDay, iHour, iMinute, iSecond, pTm->tm_zone);
return temp;
} int main()
{
    setenv("TZ","Europe/Moscow",1);
  tzset();
    time_t tt = MakeTime(2011,10,30, 2, 30, 00, -1);

    string tstring = ToString(tt);
cout<<tstring<<endl;
cout<<"Moscow exit UTC time(duplicate hour 2011-10-30 02:30:00):"<<ToString(MakeTime(2011, 10, 30, 2, 30, 00, -1))<<endl; cout<<"Isdst = -1"<<endl;
cout<<"Moscow enter UTC time(before miss hour 2011-03-27 01:30:00):"<<ToString(MakeTime(2011, 3, 27, 1, 30, 00, -1))<<endl;
cout<<"Moscow exit UTC time(duplicate hour 2011-10-30 02:30:00):"<<ToString(MakeTime(2011, 10, 30, 2, 30, 00, -1))<<endl;
cout<<"Moscow enter UTC time(miss hour 2011-03-27 02:30:00):"<<ToString(MakeTime(2011, 3, 27, 2, 30, 00, -1))<<endl;
cout<<"Moscow enter UTC time(miss hour 2011-03-27 03:30:00):"<<ToString(MakeTime(2011, 3, 27, 3, 30, 00, -1))<<endl;
cout<<"Moscow exit UTC time(before duplicate hour 2011-10-30 01:30:00):"<<ToString(MakeTime(2011, 10, 30, 1, 30, 00, -1))<<endl;
cout<<"Moscow exit UTC time(duplicate hour 2011-10-30 02:30:00):"<<ToString(MakeTime(2011, 10, 30, 2, 30, 00, -1))<<endl;
cout<<"Moscow exit UTC time(after duplicate hour 2011-10-30 03:30:00):"<<ToString(MakeTime(2011, 10 ,30, 3, 30, 00, -1))<<endl;
cout<<"Moscow enter UTC time(before miss hour 2011-03-27 01:30:00):"<<ToString(MakeTime(2011, 3, 27, 1, 30, 00, -1))<<endl;
cout<<"Moscow exit UTC time(duplicate hour 2011-10-30 02:30:00):"<<ToString(MakeTime(2011, 10, 30, 2, 30, 00, -1))<<endl; cout<<"Current time :"<<ToString(time(0))<<endl;
cout<<"Isdst = -1"<<endl;
cout<<"Moscow enter UTC time(before miss hour 2011-03-27 01:30:00):"<<ToString(MakeTime(2011, 3, 27, 1, 30, 00, -1))<<endl;
cout<<"Moscow enter UTC time(miss hour 2011-03-27 02:30:00):"<<ToString(MakeTime(2011, 3, 27, 2, 30, 00, -1))<<endl;
cout<<"Moscow enter UTC time(miss hour 2011-03-27 03:30:00):"<<ToString(MakeTime(2011, 3, 27, 3, 30, 00, -1))<<endl;
cout<<"Moscow exit UTC time(before duplicate hour 2011-10-30 01:30:00):"<<ToString(MakeTime(2011, 10, 30, 1, 30, 00, -1))<<endl;
cout<<"Moscow exit UTC time(duplicate hour 2011-10-30 02:30:00):"<<ToString(MakeTime(2011, 10, 30, 2, 30, 00, -1))<<endl;
cout<<"Moscow exit UTC time(after duplicate hour 2011-10-30 03:30:00):"<<ToString(MakeTime(2011, 10 ,30, 3, 30, 00, -1))<<endl; cout<<"Isdst = 1"<<endl;
cout<<"Moscow enter UTC time(before miss hour 2011-03-27 01:30:00):"<<ToString(MakeTime(2011, 3, 27, 1, 30, 00, 1))<<endl;
cout<<"Moscow enter UTC time(miss hour 2011-03-27 02:30:00):"<<ToString(MakeTime(2011, 3, 27, 2, 30, 00, 1))<<endl;
cout<<"Moscow enter UTC time(miss hour 2011-03-27 03:30:00):"<<ToString(MakeTime(2011, 3, 27, 3, 30, 00, 1))<<endl;
cout<<"Moscow exit UTC time(before duplicate hour 2011-10-30 01:30:00):"<<ToString(MakeTime(2011, 10, 30, 1, 30, 00, 1))<<endl;
cout<<"Moscow exit UTC time(duplicate hour 2011-10-30 02:30:00):"<<ToString(MakeTime(2011, 10, 30, 2, 30, 00, 1))<<endl;
cout<<"Moscow exit UTC time(after duplicate hour 2011-10-30 03:30:00):"<<ToString(MakeTime(2011, 10 ,30, 3, 30, 00, 1))<<endl; cout<<"Isdst = 0"<<endl;
cout<<"Moscow enter UTC time(before miss hour 2011-03-27 01:30:00):"<<ToString(MakeTime(2011, 3, 27, 1, 30, 00, 0))<<endl;
cout<<"Moscow enter UTC time(miss hour 2011-03-27 02:30:00):"<<ToString(MakeTime(2011, 3, 27, 2, 30, 00, 0))<<endl;
cout<<"Moscow enter UTC time(miss hour 2011-03-27 03:30:00):"<<ToString(MakeTime(2011, 3, 27, 3, 30, 00, 0))<<endl;
cout<<"Moscow exit UTC time(before duplicate hour 2011-10-30 01:30:00):"<<ToString(MakeTime(2011, 10, 30, 1, 30, 00, 0))<<endl;
cout<<"Moscow exit UTC time(duplicate hour 2011-10-30 02:30:00):"<<ToString(MakeTime(2011, 10, 30, 2, 30, 00, 0))<<endl;
cout<<"Moscow exit UTC time(after duplicate hour 2011-10-30 03:30:00):"<<ToString(MakeTime(2011, 10 ,30, 3, 30, 00, 0))<<endl;
}

MakeTime的测试代码

setenv("TZ","Europe/Moscow",1);
tzset();
// DST = -1

// 切入夏令时丢失的小时前的小时
assert("2011-03-27 01:30:00 MSK" == ToString(MakeTime(2011, 3, 27, 1, 30, 00, -1)));
// 切入夏令时丢失的小时
assert("2011-03-27 03:30:00 MSD" == ToString(MakeTime(2011, 3, 27, 2, 30, 00, -1)));
// 切入夏令时丢失的小时后的小时
assert("2011-03-27 03:30:00 MSD" == ToString(MakeTime(2011, 3, 27, 3, 30, 00, -1)));
// 切入夏令时丢失的小时
assert("2011-03-27 03:30:00 MSD" == ToString(MakeTime(2011, 3, 27, 2, 30, 00, -1))); // 切出夏令时重复的小时前的小时
assert("2011-10-30 01:30:00 MSD" == ToString(MakeTime(2011, 10, 30, 1, 30, 00, -1)));
// 切出夏令时重复的小时
assert("2011-10-30 02:30:00 MSD" == ToString(MakeTime(2011, 10, 30, 2, 30, 00, -1)));
// 切出夏令时重复的小时后的小时
assert("2011-10-30 03:30:00 MSK" == ToString(MakeTime(2011, 10 ,30, 3, 30, 00, -1)));
// 切出夏令时重复的小时
assert("2011-10-30 02:30:00 MSK" == ToString(MakeTime(2011, 10, 30, 2, 30, 00, -1))); // DST = 0 // 切入夏令时丢失的小时前的小时
assert("2011-03-27 01:30:00 MSK" == ToString(MakeTime(2011, 3, 27, 1, 30, 00, 0)));
// 切入夏令时丢失的小时
assert("2011-03-27 03:30:00 MSD" == ToString(MakeTime(2011, 3, 27, 2, 30, 00, 0)));
// 切入夏令时丢失的小时后的小时
assert("2011-03-27 04:30:00 MSD" == ToString(MakeTime(2011, 3, 27, 3, 30, 00, 0)));
// 切入夏令时丢失的小时
assert("2011-03-27 03:30:00 MSD" == ToString(MakeTime(2011, 3, 27, 2, 30, 00, 0))); // 切出夏令时重复的小时前的小时
assert("2011-10-30 02:30:00 MSD" == ToString(MakeTime(2011, 10, 30, 1, 30, 00, 0)));
// 切出夏令时重复的小时
assert("2011-10-30 02:30:00 MSK" == ToString(MakeTime(2011, 10, 30, 2, 30, 00, 0)));
// 切出夏令时重复的小时后的小时
assert("2011-10-30 03:30:00 MSK" == ToString(MakeTime(2011, 10 ,30, 3, 30, 00, 0)));
// 切出夏令时重复的小时
assert("2011-10-30 02:30:00 MSK" == ToString(MakeTime(2011, 10, 30, 2, 30, 00, 0))); // DST = 1 // 切入夏令时丢失的小时前的小时
assert("2011-03-27 00:30:00 MSK" == ToString(MakeTime(2011, 3, 27, 1, 30, 00, 1)));
// 切入夏令时丢失的小时
assert("2011-03-27 01:30:00 MSK" == ToString(MakeTime(2011, 3, 27, 2, 30, 00, 1)));
// 切入夏令时丢失的小时后的小时
assert("2011-03-27 03:30:00 MSD" == ToString(MakeTime(2011, 3, 27, 3, 30, 00, 1)));
// 切入夏令时丢失的小时
assert("2011-03-27 01:30:00 MSK" == ToString(MakeTime(2011, 3, 27, 2, 30, 00, 1))); // 切出夏令时重复的小时前的小时
assert("2011-10-30 01:30:00 MSD" == ToString(MakeTime(2011, 10, 30, 1, 30, 00, 1)));
// 切出夏令时重复的小时
assert("2011-10-30 02:30:00 MSD" == ToString(MakeTime(2011, 10, 30, 2, 30, 00, 1)));
// 切出夏令时重复的小时后的小时
assert("2011-10-30 02:30:00 MSK" == ToString(MakeTime(2011, 10 ,30, 3, 30, 00, 1)));
// 切出夏令时重复的小时
assert("2011-10-30 02:30:00 MSD" == ToString(MakeTime(2011, 10, 30, 2, 30, 00, 1)));

mktime 夏令时的更多相关文章

  1. mktime夏令时处理

    https://www.cnblogs.com/dongzhiquan/archive/2011/11/05/2237075.html 我们的最终目的是把字符串格式的时间转换为内部使用的“日历时间”, ...

  2. mktime性能问题调查

    一.问题提出 会议中有同学提到使用mktime遇到一些问题: 1) 设置tm_isdst后速度很慢 2) 设置TZ环境变量提速极大 所以想调查下具体情况.   mktime真的这么慢?如果是,为什么? ...

  3. C语言mktime()

    最近在调试stm32L151单片机,因为业务需要将从RTC获取的时间转换成时间戳.转换的时候发现获取的时间一直不对.一直被两个问题困扰. 1.从RTC获取出来的月份为什么比实际月份小1? 2.转换得来 ...

  4. c++ mktime()

    今天联系写一个日历的程序,需要算出月份中的第一天是星期几,用到了mktime()这个函数,感觉这个函数挺有用的,分享给大家. 原型:time_t mktime(struct tm *) 其中的tm结构 ...

  5. C语言中mktime函数功能及用法

    今天联系写一个日历的程序,需要算出月份中的第一天是星期几,用到了mktime()这个函数,感觉这个函数挺有用的,分享给大家. 原型:time_t mktime(struct tm *) 其中的tm结构 ...

  6. php date mktime 获取时间上的各种值

    echo date("Ymd",strtotime("now")), "\n"; echo date("Ymd",str ...

  7. Java中的夏令时问题

    因为在用C#做项目的时候被夏令时坑过一回,所以这次将在java中的时区转换信息做一下记录,很简单 SimpleDateFormat inputFormat = new SimpleDateFormat ...

  8. kernel/mktime

    /* *  linux/kernel/mktime.c * *  Copyright (C) 1991, 1992  Linus Torvalds */ #include <linux/mkti ...

  9. 夏令时 DST (Daylight Saving Time) java中的夏令时【转】

    1916年,德国首先实行夏令时,英国因为怕德国会从中得到更大的效益,因此紧跟着也采取了夏令时 1986年至1991年,中华人民共和国在全国范围实行了六年夏令时 サマータイム 夏時間(日本现在没有实行夏 ...

随机推荐

  1. Django分页器的设置

    Django分页器的设置 有时候在页面中数据有多条时很显然需要进行分页显示,那么在python中django可以这样设置一个分页处理 怎么样去设置呢? 我们要用到  Django  中的  Pagin ...

  2. spring注解-“@Scope”

    @Scope可用来指定Spring创建bean的方式 默认为单实例,即@Scope("singleton").IOC容器启动的时候会调用方法创建对象并放到IOC容器中,以后每次获取 ...

  3. python os.path.isfile函数

    最近刚开始学习Python,做了个小练习:扫描当前目录及其子目录中的文件,找出文件名中含有指定关键字的文件并打印文件名.思路很简单,如果是文件则判断是否满足条件:如果是目录则进入目录搜索文件,递归. ...

  4. 聊聊JMM

    JMM是什么? JMM 全称 Java memory model ,直译过来就是Java内存模型,这里注意了,指到并不是JVM中的内存分布新生代.老年代.永久代这些,当然也不是 程序计数器(PC).j ...

  5. 爬虫之scrapy扩展

    针对pipelines的扩展 from scrapy.exceptions import DropItem class CustomPipeline(object): def __init__(sel ...

  6. ThingJS平台制作第一人称视角下的巡更效果

    今天想要做一个类似巡更的过程,就像是在学校保安巡更,小区保安巡更一样.巡更需要用到相应的场景,比如说:园区.学校.超市以及工厂等等,然后需要规划好路线也就是巡更的路线,并且视角要跟随路线来变换,从而达 ...

  7. 知识点---js监听手机返回键,回到指定界面

    方法一. $(function(){ pushHistory(); window.addEventListener(“popstate”, function(e) { window.location ...

  8. unittest框架(惨不忍睹低配版)

    根据我上个随笔的unittest框架优化得来,虽然对于smtp模块还是有点迷糊,不过还是勉强搭建运行成功了,还是先上代码: #login_test.py import requests class L ...

  9. 配置错误 不能在此路径中使用此配置节。如果在父级别上锁定了该节,便会出现这种情况。锁定是默认设置的(overrideModeDefault="Deny"),或者是通过包含 overrideMode="Deny" 或旧有的 allowOverride="false" 的位置标记明确设置的。

    原因:可能是在安装IIS7的时候没有安装asp.net, 尝试使用以下方法: cmd.exe要以管理员身份启动,在c:\windows\system32下找到cmd.exe,右键管理员启动,输入命令 ...

  10. 获取select下拉框选中的的值

    有一id=test的下拉框,怎么获取选中的那个值呢? 分别使用javascript方法和jquery方法 <select id="test"  name="&quo ...