写软件时经常需要获取系统时间,然后再做各种格式的转换,这里整理了几种方法,在visual Studio 2017 中已测试通过过,以供参考。
在VS2017中,localtime()函数编译提示该函数不安全,需要替换localtime_s();
char *strDest:用来存放格式化后的字符串缓存,
size_t maxsize:指定最多可以输出的字符数,
const char *format:格式化字符串,
const struct tm *timeptr:要转换的时间。
可使用的格式化字符串:
%a 星期几的简写
%A 星期几的全称
%b 月分的简写
%B 月份的全称
%c 标准的日期的时间串
%C 年份的后两位数字
%d 十进制表示的每月的第几天
%D 月/天/年
%e 在两字符域中,十进制表示的每月的第几天
%F 年-月-日
%g 年份的后两位数字,使用基于周的年
%G 年分,使用基于周的年
%h 简写的月份名
%H 24小时制的小时
%I 12小时制的小时
%j 十进制表示的每年的第几天
%m 十进制表示的月份
%M 十时制表示的分钟数
%n 新行符
%p 本地的AM或PM的等价显示
%r 12小时的时间
%R 显示小时和分钟:hh:mm
%S 十进制的秒数
%t 水平制表符
%T 显示时分秒:hh:mm:ss
%u 每周的第几天,星期一为第一天 (值从0到6,星期一为0)
%U 第年的第几周,把星期日做为第一天(值从0到53)
%V 每年的第几周,使用基于周的年
%w 十进制表示的星期几(值从0到6,星期天为0)
%W 每年的第几周,把星期一做为第一天(值从0到53)
%x 标准的日期串
%X 标准的时间串
%y 不带世纪的十进制年份(值从0到99)
%Y 带世纪部分的十进制年份
%z,%Z 时区名称,如果不能得到时区名称则返回空字符。
方案一:localtime_s()
优点:仅使用C标准库;缺点:只能精确到秒级在VS2017中,localtime()函数编译提示该函数不安全,需要替换localtime_s();
在VS2017中 time_t是定义在corecrt.h中的一个类型,表示一个日历时间,也就是从1970年1月1日0时0分0秒到此时的秒数,原型是:
#ifndef _CRT_NO_TIME_T
#ifdef _USE_32BIT_TIME_T
typedef __time32_t time_t;
#else
typedef __time64_t time_t;
#endif
#endif
time_t 在不同系统中有不同的类型,如果是32位系统可以表示136年,也就是到2106年,64位系统估计可以表示到地老天荒~~
函数time可以获取当前日历时间时间,time的定义:
time_t time(time_t *)
time_t (typedef __int64 time_t )
time_t只是一个长整型,不符合我们的使用习惯,需要转换成本地时间,就要用到tm结构,time.h中结构tm的原型是:
//-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
//
// Types
//
//-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
struct tm
{
int tm_sec; // seconds after the minute - [0, 60] including leap second
int tm_min; // minutes after the hour - [0, 59]
int tm_hour; // hours since midnight - [0, 23]
int tm_mday; // day of the month - [1, 31]
int tm_mon; // months since January - [0, 11]
int tm_year; // years since 1900
int tm_wday; // days since Sunday - [0, 6]
int tm_yday; // days since January 1 - [0, 365]
int tm_isdst; // daylight savings time flag
};
可以看出,这个机构定义了年、月、日、时、分、秒、星期、当年中的某一天、夏令时。可以用这个结构很方便的显示时间。
使用localtime_s获取当前系统时间,该函数将一个time_t时间转换成tm结构表示的时间
使用gmtime_s函数获取格林尼治时间,在VS2017中他们的函数原型为:
static __inline errno_t __CRTDECL localtime_s(
_Out_ struct tm* const _Tm,
_In_ time_t const* const _Time
)
{
return _localtime64_s(_Tm, _Time);
}
static __inline errno_t __CRTDECL gmtime_s(
_Out_ struct tm* const _Tm,
_In_ time_t const* const _Time
)
{
return _gmtime64_s(_Tm, _Time);
}
输出方式1:
#include <stdio.h>
#include <time.h>
void dsptime(const struct tm * ptm)
{
printf("%d-%d-%d \n", (ptm->tm_year + 1900), (ptm->tm_mon + 1), ptm->tm_mday);
printf("%d-%d-%d \n", ptm->tm_hour, ptm->tm_min, ptm->tm_sec);
};
int main(void)
{
/**获取日历时间**/
time_t nowtime;
nowtime = time(NULL);
printf("nowtime = %lld \n", nowtime);
/**获取当前系统时间**/
struct tm local;
localtime_s(&local, &nowtime);
dsptime(&local);
/**获取格林威治时间**/
printf("\n");
gmtime_s(&local, &nowtime);
dsptime(&local);
getchar();
return 0;
}
输出方式2:
#include <stdio.h>
#include <time.h>
void dsptime(const struct tm * ptm)
{
printf("%d-%d-%d \n", (ptm->tm_year + 1900), (ptm->tm_mon + 1), ptm->tm_mday);
printf("%d-%d-%d \n", ptm->tm_hour, ptm->tm_min, ptm->tm_sec);
};
int main(void)
{
char tmp[64];
time_t nowtime = time(0);
struct tm local;
localtime_s(&local, &nowtime);
strftime(tmp, sizeof(tmp), "%Y/%m/%d %X %A 本年第%j天 %z", &local);
puts(tmp);
getchar();
return 0;
}
C/C++在time.h中提供了一个自定义时间格式的函数strftime,函数原型:
size_t strftime(char *strDest, size_t maxsize, const char *format, const struct tm *timeptr);
参数说明:
char *strDest:用来存放格式化后的字符串缓存,
size_t maxsize:指定最多可以输出的字符数,
const char *format:格式化字符串,
const struct tm *timeptr:要转换的时间。
可使用的格式化字符串:
%a 星期几的简写
%A 星期几的全称
%b 月分的简写
%B 月份的全称
%c 标准的日期的时间串
%C 年份的后两位数字
%d 十进制表示的每月的第几天
%D 月/天/年
%e 在两字符域中,十进制表示的每月的第几天
%F 年-月-日
%g 年份的后两位数字,使用基于周的年
%G 年分,使用基于周的年
%h 简写的月份名
%H 24小时制的小时
%I 12小时制的小时
%j 十进制表示的每年的第几天
%m 十进制表示的月份
%M 十时制表示的分钟数
%n 新行符
%p 本地的AM或PM的等价显示
%r 12小时的时间
%R 显示小时和分钟:hh:mm
%S 十进制的秒数
%t 水平制表符
%T 显示时分秒:hh:mm:ss
%u 每周的第几天,星期一为第一天 (值从0到6,星期一为0)
%U 第年的第几周,把星期日做为第一天(值从0到53)
%V 每年的第几周,使用基于周的年
%w 十进制表示的星期几(值从0到6,星期天为0)
%W 每年的第几周,把星期一做为第一天(值从0到53)
%x 标准的日期串
%X 标准的时间串
%y 不带世纪的十进制年份(值从0到99)
%Y 带世纪部分的十进制年份
%z,%Z 时区名称,如果不能得到时区名称则返回空字符。
%% 百分号
方案二:GetLocalTime( &sys );
优点:能精确到毫秒级;缺点:使用了windows API
#include <windows.h>
#include <stdio.h>
int main(void)
{
SYSTEMTIME sys;
GetLocalTime(&sys);
printf("%4d/%02d/%02d %02d:%02d:%02d.%03d 星期%1d/n", sys.wYear, sys.wMonth, sys.wDay, sys.wHour, sys.wMinute, sys.wSecond, sys.wMilliseconds, sys.wDayOfWeek);
getchar();
return 0;
}
方案三:system("time");
优点:利用系统函数,还能修改系统时间
#include<stdlib.h>
#include<iostream>
using namespace std;
void main()
{
system("time");
}
方案四:time(null)
将当前时间折算为秒级,再通过相应的时间换算即可
#include<stdio.h>
#include<time.h>
int main()
{
time_t now_time;
now_time = time(NULL);
printf("now_time = %lld \n", now_time);
getchar();
return 0;
}
以上内容根据网上资料整理而来,上面代码都已在VS2017上测试通过。