ACM-时区间时间的转换

问题描述

        直到 19 世纪,时间校准是一个纯粹的地方现象。每一个村庄当太阳升到最高点的时候把他们的时钟调到中午 12 点。一个钟表制造商人家或者村里主表的时间被认为是官方时间,市民们把自家的钟表和这个时间对齐。每周一些热心的市民会带着时间标准的表,游走大街小巷为其他市民对表。在城市之间旅游的话,在到达新地方的时候需要把怀表校准。但是,当铁路投入使用之后,越来越多的人频繁地长距离地往来,时间变得越来越重要。在铁路的早期,时刻表非常让人迷惑,每一个所谓的停靠时间都是基于停靠地点的当地时间。时间的标准化对于铁路的高效运营变得非常重要。
        在 1878 年,加拿大人 Sir Sanford Fleming 提议使用一个全球的时区(这个建议被采纳,并衍生了今天我们所使用的全球时区的概念),他建议把世界分成 24 个时区,每一个跨越 15 度经线(因为地球的经度 360 度,划分成 24 块后,一块为 15 度)。Sir SanfordFleming 的方法解决了一个全球性的时间混乱的问题。
        美国铁路公司于 1883 年 11 月 18 日使用了 Fleming 提议的时间方式。 1884 年一个国际子午线会议在华盛顿召开,他的目的是选择一个合适的本初子午线。大会最终选定了格林威治为标准的 0 度。尽管时区被确定了下来,但是各个国家并没有立刻更改他们的时间规范,在美国,尽管到 1895 年已经有很多州开始使用标准时区时间,国会直到1918 年才强制使用会议制定的时间规范。
        今天各个国家使用的是一个 Fleming 时区规范的一个变种,中国一共跨越了 5 个时区,但是使用了一个统一的时间规范,比 Coordinated Universal Time( UTC,格林威制时间)早 8 个小时。俄罗斯也拥护这个时区规范,尽管整个国家使用的时间和标准时区提前了 1 个小时。
        澳大利亚使用 3 个时区,其中主时区提前于他按 Fleming 规范的时区半小时。很多中东国家也使用了半时时区(即不是按照 Fleming 的 24 个整数时区)。
        因为时区是对经度进行划分,在南极或者北极工作的科学家直接使用了 UTC 时间,否则南极大陆将被分解成 24 个时区。
        时区的转化表如下:

UTC Coordinated Universal Time
GMT Greenwich Mean Time, 定义为 UTC
BST British Summer Time, 定义为 UTC+1 hour
IST Irish Summer Time, 定义为 UTC+1 hour
WET Western Europe Time, 定义为 UTC
WEST Western Europe Summer Time, 定义为 UTC+1 hour
CET Central Europe Time, 定义为 UTC+1
CEST Central Europe Summer Time, 定义为 UTC+2
EET Eastern Europe Time, 定义为 UTC+2
EEST Eastern Europe Summer Time, 定义为 UTC+3
MSK Moscow Time, 定义为 UTC+3
MSD Moscow Summer Time, 定义为 UTC+4
AST Atlantic Standard Time, 定义为 UTC-4 hours
ADT Atlantic Daylight Time, 定义为 UTC-3 hours
NST Newfoundland Standard Time, 定义为 UTC-3.5 hours
NDT Newfoundland Daylight Time, 定义为 UTC-2.5 hours
EST Eastern Standard Time, 定义为 UTC-5 hours
EDT Eastern Daylight Saving Time, 定义为 UTC-4 hours
CST Central Standard Time, 定义为 UTC-6 hours
CDT Central Daylight Saving Time, 定义为 UTC-5 hours
MST Mountain Standard Time, 定义为 UTC-7 hours
MDT Mountain Daylight Saving Time, 定义为 UTC-6 hours
PST Pacific Standard Time, 定义为 UTC-8 hours
PDT Pacific Daylight Saving Time, 定义为 UTC-7 hours
HST Hawaiian Standard Time, 定义为 UTC-10 hours
AKST Alaska Standard Time, 定义为 UTC-9 hours
AKDT Alaska Standard Daylight Saving Time, 定义为 UTC-8 hours
AEST Australian Eastern Standard Time, 定义为 UTC+10 hours
AEDT Australian Eastern Daylight Time, 定义为 UTC+11 hours
ACST Australian Central Standard Time, 定义为 UTC+9.5 hours
ACDT Australian Central Daylight Time, 定义为 UTC+10.5 hours
AWST Australian Western Standard Time, 定义为 UTC+8 hours

下面给出了一些时间,请在不同时区之间进行转化。

输入数据

       输入的第一行包含了一个整数 N,表示有 N 组测试数据。接下来 N 行,每一行包括一个时间和两个时区的缩写,它们之间用空格隔开。时间由标准的 a.m./p.m 给出。midnight 表示晚上 12 点( 12:00 a.m.), noon 表示中午 12 点( 12:00 p.m.)。

输出要求

       假设输入行给出的时间是在第一个时区中的标准时间,要求输出这个时间在第二个时区中的标准时间。

输入样例

4
noon HST CEST
11:29 a.m. EST GMT
6:01 p.m. CST UTC
12:40 p.m. ADT MSK

输出样例

midnight
4:29 p.m.
12:01 a.m.
6:40 p.m.

解题思路

        这个题目要求在两个时区之间进行时间的转换。我们根据每个时区与格林威治时间的转换公式可以推算出两个时区之间的差别。问题的解决方法不难想到,只是日期处理类问题具有共同的特点就是输入输出比较麻烦,有一些需要特殊处理的情况,例如转换后多出一天或少了一天的情况需要处理。具体到这个题目来说:输入时,除了一般的时间表示法:时:分a.m/p.m.之外,要特殊处理 noon 和 midnight;在直接通过格林威治时间进行转换后,要判断是否超过一天或减少了一天的情况;在输出时间时,要对 noon 和 midnight 进行特殊处理。
        解决这个问题时,关键的是确定两个时区之间的时差。因为时区是用字符串形式给出的,所以要先将时区对应到该时区与格林威治时间的时差上。有了每个时区与格林威治时间的时差,就可以计算任意两个时区之间的时差。

参考程序

#include <iostream>
#include <cstdio>
#include <cstring>
#include <string>
using namespace std;
int difference(char* zone1,char* zone2){// 计算两个时区之间的时差,以分钟为单位。
	char* zone[32]={"UTC","GMT","BST","IST","WET","WEST",
					"CET","CEST","EET","EEST","MSK",
					"MSD","AST","ADT","NST","NDT",
					"EST","EDT","CST","CDT","MST",
					"MDT","PST","PDT","HST","AKST",
					"AKDT","AEST","AEDT","ACST","ACDT",
					"AWST"};
	float time[32]={0,0,1,1,0,1,1,2,2,3,3,4,-4,-3,-3.5,-2.5,-5,-4,-6,-5,-7,
					-6,-8,-7,-10,-9,-8,10,11,9.5,10.5,8};
	int i,j;
	for(i=0;strcmp(zone[i],zone1);i++);//找到第一个时区对应的位置
	for(j=0;strcmp(zone[j],zone2);j++);//找到第二个时区对应的位置
	return (int)((time[i] - time[j])*60);//计算并返回时差,以分钟为单位
}
int main()
{
	int nCases;
	cin>>nCases;// 读入测试数据数目
	for(int i=0;i<nCases;i++){// 对每组输入数据
		char time[9];// 输入的时间
		int hours=0,minutes=0;// 转换成整数
		cin>>time; //读入时间
		switch(time[0]){
			case 'n':// 输入为”noon”
					hours=12;
					break;
			case 'm':	// 输入为”midnight”
					break;
			default:
				sscanf(time,"%d:%d",&hours,&minutes);// 输入为 时:分
				hours %= 12;
				cin>>time;
				if(time[0] == 'p')
					hours += 12;
		}
		char timezone1[5],timezone2[5];
		cin>>timezone1>>timezone2;// 读入时区
		int newTime;//以分钟为单位
		newTime = hours * 60 + minutes + difference(timezone2,timezone1);
		if(newTime < 0)
			newTime += 1440;// 提前一天,将负的时间加上一天。
		newTime %= 1440;// 如果超过一天,将一天的时间减去。
		switch(newTime){
			case 0:
				cout<<"midnight"<<endl;// 新时间为凌晨
				break;
			case 720:
				cout<<"noon"<<endl;// 新时间为中午
				break;
			default:
				hours = newTime / 60;
				minutes = newTime % 60;
				if(hours == 0){// 凌晨, 分钟不为 0
					printf("12:%02d a.m.\n",minutes);
				}else if(hours < 12){// 上午
					printf("%d:%02d p.m.\n",hours,minutes);
				}else if(hours == 12){// 中午, 分钟不为 0
					printf("12:%02d p.m.\n", minutes);
				}else{// 下午
					printf("%d:%02d p.m.\n", hours%12, minutes);
				}
		}
	}
	return 0;
}


  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值