现在的位置: 首页 > 综合 > 正文

C语言里时间函数的操作

2012年09月09日 ⁄ 综合 ⁄ 共 4119字 ⁄ 字号 评论关闭

导读:
  Calendar Time:日历时间,是用“从一个标准时间点到此时的时间经过的秒数”来表示的时间。这个标准时间点对不同的编译器来说会有所不同,但对一个编译系统来说,这个标准时间点是不变的,该编译系统中的时间对应的日历时间都通过该标准时间点来衡量,所以可以说日历时间是“相对时间”,但是无论你在哪一个时区,在同一时刻对同一个标准时间点来说,日历时间都是一样的。
  在C语言里的时间我们可以这样理解,我们先获取日历时间,存放日历时间应该是一个long型的数据.即time(NULL)的返回值就是long型的数据.
  C语言里的有一个结构体struct tm
  #ifndef _TM_DEFINED /*如果没有定义_TM_DEFINED*/
  struct tm {
  int tm_sec; /* 秒 – 取值区间为[0,59] */
  int tm_min; /* 分 - 取值区间为[0,59] */
  int tm_hour; /* 时 - 取值区间为[0,23] */
  int tm_mday; /* 一个月中的日期 - 取值区间为[1,31] */
  int tm_mon; /* 月份(从一月开始,0代表一月) - 取值区间为[0,11] */
  int tm_year; /* 年份,其值等于实际年份减去1900 */
  int tm_wday; /* 星期 – 取值区间为[0,6],其中0代表星期天,1代表星期一,以此类推 */
  int tm_yday; /* 从每年的1月1日开始的天数 – 取值区间为[0,365],其中0代表1月1日,1代表1月2日,以此类推 */
  int tm_isdst; /* 夏令时标识符,实行夏令时的时候,tm_isdst为正。不实行夏令时的进候,tm_isdst为0;不了解情况时,tm_isdst()为负。*/
  };
  #define _TM_DEFINED /*防止struct tm重复定义*/
  #endif
  看完上面的结构体后,是不是感觉时间操作相对简单多了,不需要理会复杂的日历时间,只要利用函数把日历时间填充到struct tm即可
  哪些函数能完成上述功能呢?
  struct tm * gmtime(const time_t *timer); /*将日历时间填入后,转换格林威治时间*/
  struct tm * localtime(const time_t * timer); /*将日历时间填入结构体,是本地时间*/
  比如我们只想做些简单的打印,而并非完全定制,即它们的返回值是字符串
  char * asctime(const struct tm * timeptr); /*函数的形参为struct tm的指针
  char * ctime(const time_t *timer); /*函数的形参为日历时间*/
  时间的格式完全自定义
  实现该功能的函数strftime(),它的原型为
  size_t strftime(
  char *strDest,
  size_t maxsize,
  const char *format,
  const struct tm *timeptr
  );
  下面是该函数的一些基本操作
  函数strftime()的操作有些类似于sprintf():识别以百分号(%)开始的格式命令集合,格式化输出结果放在一个字符串中。格式化命令说明串strDest中各种日期和时间信息的确切表示方法。格式串中的其他字符原样放进串中。格式命令列在下面,它们是区分大小写的。
  %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 时区名称,如果不能得到时区名称则返回空字符。
  %% 百分号
  但是经验证,并不是每个选项都可以用.
  附一个例子
  下面的程序则显示当前的完整日期:
  #include
  #include
  void main( void )
  {
  struct tm *newtime;
  char tmpbuf[128];
  time_t lt1;
  time(   newtime=localtime(  strftime( tmpbuf, 128, "Today is %A, day %d of %B in the year %Y./n", newtime);
  printf(tmpbuf);
  }
  运行结果:
  Today is Saturday, day 30 of July in the year 2005.
  计算持续时间的长度
  1)
  有时候在实际应用中要计算一个事件持续的时间长度,比如计算打字速度。在第1节计时部分中,我已经用clock函数举了一个例子。Clock()函数可以精确到毫秒级。同时,我们也可以使用difftime()函数,但它只能精确到秒。该函数的定义如下:
  double difftime(time_t time1, time_t time0);
  虽然该函数返回的以秒计算的时间间隔是double类型的,但这并不说明该时间具有同double一样的精确度,这是由它的参数觉得的(time_t是以秒为单位计算的)。比如下面一段程序:
  #include "time.h"
  #include "stdio.h"
  #include "stdlib.h"
  int main(void)
  {
  time_t start,end;
  start = time(NULL);
  system("pause");
  end = time(NULL);
  printf("The pause used %f seconds./n",difftime(end,start));//  system("pause");
  return 0;
  }
  运行结果为:
  请按任意键继续. . .
  The pause used 2.000000 seconds.
  请按任意键继续. . .
  可以想像,暂停的时间并不那么巧是整整2秒钟。其实,你将上面程序的带有“//  printf("The pause used %f seconds./n",end-start);
  2)
  #include “stdio.h”
  #include “stdlib.h”
  #include “time.h”
  int main( void )
  {
  long i = 10000000L;
  clock_t start, finish;
  double duration;
  /* 测量一个事件持续的时间*/
  printf( "Time to do %ld empty loops is ", i );
  start = clock();
  while( i-- ) ;
  finish = clock();
  duration = (double)(finish - start) / CLOCKS_PER_SEC;
  printf( "%f seconds/n", duration );
  system("pause");
  }
  在笔者的机器上,运行结果如下:
  Time to do 10000000 empty loops is 0.03000 seconds
  上面我们看到时钟计时单元的长度为1毫秒,那么计时的精度也为1毫秒,那么我们可不可以通过改变CLOCKS_PER_SEC的定义,通过把它定义的大一些,从而使计时精度更高呢?通过尝试,你会发现这样是不行的。在标准C/C++中,最小的计时单位是一毫秒。
  分解时间转化为日历时间
  这里说的分解时间就是以年、月、日、时、分、秒等分量保存的时间结构,在C/C++中是tm结构。我们可以使用mktime()函数将用tm结构表示的时间转化为日历时间。其函数原型如下:
  time_t mktime(struct tm * timeptr);
  其返回值就是转化后的日历时间。这样我们就可以先制定一个分解时间,然后对这个时间进行操作了,下面的例子可以计算出1997年7月1日是星期几:
  #include "time.h"
  #include "stdio.h"
  #include "stdlib.h"
  int main(void)
  {
  struct tm t;
  time_t t_of_day;
  t.tm_year=1997-1900;
  t.tm_mon=6;
  t.tm_mday=1;
  t.tm_hour=0;
  t.tm_min=0;
  t.tm_sec=1;
  t.tm_isdst=0;
  t_of_day=mktime(&t);
  printf(ctime(&t_of_day));
  return 0;
  }
  运行结果:
  Tue Jul 01 00:00:01 1997
  现在注意了,有了mktime()函数,是不是我们可以操作现在之前的任何时间呢?你可以通过这种办法算出1945年8月15号是星期几吗?答案是否定的。因为这个时间在1970年1月1日之前,所以在大多数编译器中,这样的程序虽然可以编译通过,但运行时会异常终止。

本文转自
http://blog.cfan.com.cn/html/88/10088_itemid_57440.html

抱歉!评论已关闭.