版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
C语言中如何获取时间?精度如何?1使用time_ttime(time_t*timer)精确到秒2使用clock_tclock()得到的是CPU时间精确到1/CLOCKS_PER_SEC秒3计算时间差使用doubledifftime(time_ttimer1,time_ttimer0)4使用DWORDGetTickCount()精确到毫秒5如果使用MFC的CTime类,可以用CTime::GetCurrentTime()精确到秒6要获取高精度时间,可以使用BOOLQueryPerformanceFrequency(LARGE_INTEGER*lpFrequency)获取系统的计数器的频率BOOLQueryPerformanceCounter(LARGE_INTEGER*lpPerformanceCount)获取计数器的值然后用两次计数器的差除以Frequency就得到时间。7MultimediaTimerFunctionsThefollowingfunctionsareusedwithmultimediatimers.timeBeginPeriod/timeEndPeriod/timeGetDevCaps/timeGetSystemTime//*********************************************************************//用标准C实现获取当前系统时间的函数一.time()函数time(&rawtime)函数获取当前时间距1970年1月1日的秒数,以秒计数单位,存于rawtime中。#include"time.h"voidmain()(time_trawtime;structtm*timeinfo;time(&rawtime);timeinfo=localtime(&rawtime);printf("\007Thecurrentdate/timeis:%s”,asctime(timeinfo));exit(0);}=================#include--必须的时间函数头文件time_t--时间类型(time.h定义是typedeflongtime_t;追根溯源,time_t是long)structtm--时间结构,time.h定义如下:inttm_sec;inttm_min;inttm_hour;inttm_mday;inttm_mon;inttm_year;inttm_wday;inttm_yday;inttm_isdst;time(&rawtime);--获取时间,以秒计,从1970年1月一日起算,存于rawtimelocaltime(&rawtime);--转为当地时间,tm时间结构asctime()--转为标准ASCII时间格式:星期月日时:分:秒年二.clock()函数,用clock()函数,得到系统启动以后的毫秒级时间,然后除以CLOCKS_PER_SEC,就可以换成“秒”,标准c函数。clock_tclock(void);#includeclock_tt=clock();longsec=t/CLOCKS_PER_SEC;他是记录时钟周期的,实现看来不会很精确,需要试验验证;三.gettime(&t);据说tc2.0的time结构含有毫秒信息#include#includeintmain(void)(structtimet;gettime(&t);printf("Thecurrenttimeis:-:d:d.d\n”,t.ti_hour,t.ti_min,t.ti_sec,t.ti_hund);return0;}time是一个结构体,,其中成员函数ti_hund是毫秒。。。四.GetTickCount(),这个是windows里面常用来计算程序运行时间的函数;DWORDdwStart=GetTickCount();〃这里运行你的程序代码DWORDdwEnd=GetTickCount();则(dwEnd-dwStart)就是你的程序运行时间,以毫秒为单位这个函数只精确到55ms,1个tick就是55ms。五.timeGetTime()t,imeGetTime()基本等于GetTickCount(),但是精度更高DWORDdwStart=timeGetTime();〃这里运行你的程序代码DWORDdwEnd=timeGetTime();则(dwEnd-dwStart)就是你的程序运行时间,以毫秒为单位虽然返回的值单位应该是ms,但传说精度只有10ms。〃*****************************************************************Unix##unix时间相关,也是标准库的//*********************************************************************timegm函数只是将structtm结构转成time_t结构,不使用时区信息;time_ttimegm(structtm*tm);mktime使用时区信息time_tmktime(structtm*tm);timelocal函数是GNU扩展的与posix函数mktime相当time_ttimelocal(structtm*tm);3.gmtime函数只是将time_t结构转成structtm结构,不使用时区信息;structtm*gmtime(consttime_t*clock);4.localtime使用时区信息structtm*localtime(consttime_t*clock);time获取时间,stime设置时间time_tt;t=time(&t);stime其参数应该是GMT时间,根据本地时区设置为本地时间;intstime(time_t*tp)UTC=true表示采用夏时制;文件的修改时间等信息全部采用GMT时间存放,不同的系统在得到修改时间后通过localtime转换成本地时间;设置时区推荐使用setup来设置;6.设置时区也可以先更变/etc/sysconfig/clock中的设置再将ln-fs/usr/share/zoneinfo/xxxx/xxx/etc/localtime才能重效time_t只能表示68年的范围,即mktime只能返回1970-2038这一段范围的time_t看看你的系统是否有time_t64,它能表示更大的时间范围〃***************************************************************windows##Window里面的一些不一样的//*********************************************************************CTime()类VC编程一般使用CTime类获得当前日期和时间CTimet=GetCurrentTime();SYSTEMTIME结构包含毫秒信息typedefstruct_SYSTEMTIME(WORDwYear;WORDwMonth;WORDwDayOfWeek;WORDwDay;WORDwHour;WORDwMinute;WORDwSecond;WORDwMilliseconds;}SYSTEMTIME,*PSYSTEMTIME;SYSTEMTIMEt1;GetSystemTime(&t1)CTimecurTime(t1);WORDms=t1.wMilliseconds;SYSTEMTIMEsysTm;::GetLocalTime(&sysTm);在time.h中的_strtime()〃只能在windows中用chart[11];_strtime(t);puts(t);〃*****************************获得当前日期和时间CTimetm=CTime::GetCurrentTime();CStringstr=tm.Format("%Y-%m-%d");在VC中,我们可以借助CTime时间类,获取系统当前日期,具体使用方法如下:CTimet=CTime::GetCurrentTime();〃获取系统日期,存储在t里面intd=t.GetDay();〃获得当前日期inty=t.GetYear();//获取当前年份intm=t.GetMonth();〃获取当前月份inth=t.GetHour();〃获取当前为几时intmm=t.GetMinute();//获取当前分钟ints=t.GetSecond();〃获取当前秒intw=t.GetDayOfWeek();〃获取星期几,注意1为星期天,7为星期六CTimeSpan类如果想计算两段时间的差值,可以使用CTimeSpan类,具体使用方法如下:CTimet1(1999,3,19,22,15,0);CTimet=CTime::GetCurrentTime();CTimeSpanspan=t-t1;〃计算当前系统时间与时间t1的间隔intiDay=span.GetDays();〃获取这段时间间隔共有多少天intiHour=span.GetTotalHours();〃获取总共有多少小时intiMin=span.GetTotalMinutes();//获取总共有多少分钟intiSec=span.GetTotalSeconds();//获取总共有多少秒_timeb()函数_timeb定义在SYS\TIMEB.H,有四个fieldsdstflagmillitmtimetimezonevoid_ftime(struct_timeb*timeptr);struct_timebtimebuffer;_ftime(&timebuffer);取当前时间:文档讲可以到ms,有人测试,好象只能到16ms!设置计时器定义TIMERID#defineTIMERID_JISUANFANGSHI2在适当的地方设置时钟,需要开始其作用的地方;SetTimer(TIMERID_JISUANFANGSHI,200,NULL);在不需要定时器的时候的时候销毁掉时钟KillTimer(TIMERID_JISUANFANGSHI);对应VC程序的消息映射voidCJisuan::OnTimer(UINTnIDEvent)(switch(nlDEvent)}—##如何设定当前系统时间windowsSYSTEMTIMEm_myLocalTime,*lpSystemTime;m_myLocalTime.wYear=2003;m_myLocalTime.wM;m_myLocalTime.wDay=1;m_myLocalTime.wHour=0;m_myLocalTime.wMinute=0;m_myLocalTime.wSec;m_myLocalTime.wMillisec;lpSystemTime=&m_myLocalTime;if(SetLocalTime(lpSystemTime))//此处换成SetSystemTime()也不行MessageBox("OK!");elseMessageBox("Error!");SYSTEMTIMEm_myLocalTime,*lpSystemTime;m_myLocalTime.wYear=2003;m_myLocalTime.wM;m_myLocalTime.wDay=1;lpSystemTime=&m_myLocalTime;if(SetDate(lpSystemTime))〃此处换成SetSystemTime()也不行MessageBox("OK!");elseMessageBox("Error!");本文来自CSDN博客,转载请标明出处:/khuang2008/archive/2008/12/09/3483274.aspx一种制作微秒级精度定时器的方法当使用定时器时,在很多情况下只用到毫秒级的时间间隔,所以只需用到下面的两种常用方式就满足要求了。一是用SetTimer函数建立一个定时器后,在程序中通过处理由定时器发送到线程消息队列中的WM_TIMER消息,而得到定时的效果(退出程序时别忘了调用和SetTimer配对使用的KillTimer函数)。二是利用GetTickCount函数可以返回自计算机启动后的时间,通过两次调用GetTickCount函数,然后控制它们的差值来取得定时效果,此方式跟第一种方式一样,精度也是毫秒级的。用这两种方式取得的定时效果虽然在许多场合已经满足实际的要求,但由于它们的精度只有毫秒级的,而且在要求定时时间间隔小时,实际定时误差大。下面介绍一种能取得高精度定时的方法。在一些计算机硬件系统中,包含有高精度运行计数器(high-resolutionperformancecounter),利用它可以获得高精度定时间隔,其精度与CPU的时钟频率有关。采用这种方法的步骤如下:1、首先调用QueryPerformanceFrequency函数取得高精度运行计数器的频率f。单位是每秒多少次(n/s),此数一般很大。2、在需要定时的代码的两端分别调用QueryPerformanceCounter以取得高精度运行计数器的数值n1,n2。两次数值的差值通过f换算成时间间隔,t=(n2-n1)/f。下面举一个例子来演示这种方法的使用及它的精确度。在VC6.0下用MFC建立一个对话框工程,取名为HightTimer.在对话框面板中控件的布局如下图:其中包含两个静态文本框,两个编辑框和两个按纽。上面和下面位置的编辑框的ID分别为IDC_E_TEST和IDC_E_ACTUAL,通过MFCClassWizard添加的成员变量也分别对应为DWORDm_dwTest和DWORDm_dwAct.“退出”按纽的ID为IDOK,"开始测试”按纽ID为IDC_B_TEST,用MFCClassWizard添加此按纽的单击消息处理函数如下:voidCHightTimerDlg::OnBTest()(//TODO:AddyourcontrolnotificationhandlercodehereUpdateData(TRUE);//取输入的测试时间值到与编辑框相关联的成员变量m_dwTest中LARGE_INTEGERfrequence;if(!QueryPerformanceFrequency(&frequence))//取高精度运行计数器的频率,若硬件不支持则返回FALSEMessageBox("Yourcomputerhardwaredoesn'tsupportthehigh-resolutionperformancecounter","NotSupport",MB_ICONEXCLAMATION|MB_OK);LARGE_INTEGERtest,ret;test.QuadPart=frequence.QuadPart*m_dwTest/1000000;〃通过频率换算微秒数到对应的数量(与CPU时钟有关),1秒=1000000微秒ret=MySleep(test);〃调用此函数开始延时,返回实际花销的数量m_dwAct=(DWORD)(1000000*ret.QuadPart/frequence.QuadPart);//换算到微秒数UpdateData(FALSE);//显示到对话框面板}其中上面调用的MySleep函数如下:LARGE_INTEGERCHightTimerDlg::MySleep(LARGE_INTEGERInterval)///////////////////////////////////////////////////////////////////////////////////////////////////////////////功能:执行实际的延时功能//参数:Interval参数为需要执行的延时与时间有关的数量//返回值:返回此函数执行后实际所用的时间有关的数量///////////////////////////////////////////////////////////////////////////////////////////////////////////(LARGE_INTEGERprivious,current,Elapse;QueryPerformanceCounter(&privious);current=privious;while(current.QuadPart-privious.QuadPart<Interval.QuadPart)QueryPerformanceCounter(at);Elapse.QuadPart=current.QuadPart-privious.QuadPart;returnElapse;}注:别忘了在头文件中为此函数添加函数声明。至此,可以编译和执行此工程了,结果如上图所示。在本人所用的机上(奔腾366,64M内存)测试,当测试时间超过3微秒时,准确度已经非常高了,此时机器执行本身延时函数代码的时间对需要延时的时间影响很小了。上面的函数由于演示测试的需要,没有在函数级封装,下面给出的函数基本上可以以全局函数的形式照搬到别的程序中。BOOLMySleep(DWORDdwInterval)///////////////////////////////////////////////////////////////////////////////////////////////////////////////功能:执行微秒级的延时功能//参数:Interval参数为需要的延时数(单位:微秒)//返回值:若计算机硬件不支持此功能,返回FALSE,若函数执行成功,返回TRUE///////////////////////////////////////////////////////////////////////////////////////////////////////////(BOOLbNormal=TRUE;LARGE_INTEGERfrequence,privious,current,interval;if(!QueryPerformanceFrequency(&frequence))(::MessageBox(NULL,"Yourcomputerhardwaredoesn'tsupportthehigh-resolutionperformancecounter","NotSupport",MB_ICONEXCLAMATION|MB_OK);//或其它的提示信息returnFALSE;interval.QuadPart=frequence.QuadPart*dwInterval/1000000;bNormal=bNormal&&QueryPerformanceCounter(&privious);current=privious;while(current.QuadPart-privious.QuadPart<interval.QuadPart)bNormal=bNormal&&QueryPerformanceCounter(口t);returnbNormal;}需要指出的是,由于在此函数中的代码很多,机器在执行这些代码所花费的时间也很长,所以在需要几个微秒的延时时,会影响精度。实际上,读者在熟悉这种方法后,只要使用QueryPerformanceFrequency和QueryPerformanceCounter这两个函数就能按实际需要写出自己的延时代码了。使用CPU时间戳进行高精度计时对关注性能的程序开发人员而言,一个好的计时部件既是益友,也是良师。计时器既可以作为程序组件帮助程序员精确的控制程序进程,又是一件有力的调试武器,在有经验的程序员手里可以尽快的确定程序的性能瓶颈,或者对不同的算法作出有说服力的性能比较。在Windows平台下,常用的计时器有两种,一种是timeGetTime多媒体计时器,它可以提供毫秒级的计时。但这个精度对很多应用场合而言还是太粗糙了。另一种是QueryPerformanceCount计数器,随系统的不同可以提供微秒级的计数。对于实时图形处理、多媒体数据流处理、或者实时系统构造的程序员,善用QueryPerformanceCount/QueryPerformanceFrequency是一项基本功。本文要介绍的,是另一种直接利用PentiumCPU内部时间戳进行计时的高精度计时手段。以下讨论主要得益于《Windows图形编程》一书,第15页一17页,有兴趣的读者可以直接参考该书。关于RDTSC指令的详细讨论,可以参考Intel产品手册。本文仅仅作抛砖之用。在IntelPentium以上级别的CPU中,有一个称为“时间戳(TimeStamp)”的部件,它以64位无符号整型数的格式,记录了自CPU上电以来所经过的时钟周期数。由于目前的CPU主频都非常高,因此这个部件可以达到纳秒级的计时精度。这个精确性是上述两种方法所无法比拟的。在Pentium以上的CPU中,提供了一条机器指令RDTSC(ReadTimeStampCounter)来读取这个时间戳的数字,并将其保存在EDX:EAX寄存器对中。由于EDX:EAX寄存器对恰好是Win32平台下C++语言保存函数返回值的寄存器,所以我们可以把这条指令看成是一个普通的函数调用。像这样:inlineunsigned__int64GetCycleCount()(__asmRDTSC}但是不行,因为RDTSC不被C++的内嵌汇编器直接支持,所以我们要用_emit伪指令直接嵌入该指令的机器码形式0X0F、0X31,如下:inlineunsigned__int64GetCycleCount()(__asm_emit0x0F__asm_emit0x31}以后在需要计数器的场合,可以像使用普通的Win32API一样,调用两次GetCycleCount函数,比较两个返回值的差,像这样:unsignedlongt;t=(unsignedlong)GetCycleCount();//DoSomethingtime-intensive...t-=(unsignedlong)GetCycleCount();《Windows图形编程》第15页编写了一个类,把这个计数器封装起来。有兴趣的读者可以去参考那个类的代码。作者为了更精确的定时,做了一点小小的改进,把执行RDTSC指令的时间,通过连续两次调用GetCycleCount函数计算出来并保存了起来,以后每次计时结束后,都从实际得到的计数中减掉这一小段时间,以得到更准确的计时数字。但我个人觉得这一点点改进意义不大。在我的机器上实测,这条指令大概花掉了几十到100多个周期,在Celeron800MHz的机器上,这不过是十分之一微秒的时间。对大多数应用来说,这点时间完全可以忽略不计;而对那些确实要精确到纳秒数量级的应用来说,这个补偿也过于粗糙了。这个方法的优点是:高精度。可以直接达到纳秒级的计时精度(在1GHz的CPU上每个时钟周期就是一纳秒),这是其他计时方法所难以企及的。成本低。timeGetTime函数需要链接多媒体库winmm.lib,QueryPerformance*函数根据MSDN的说明,需要硬件的支持(虽然我还没有见过不支持的机器)和KERNEL库的支持,所以二者都只能在Windows平台下使用(关于DOS平台下的高精度计时问题,可以参考《图形程序开发人员指南》,里面有关于控制定时器8253的详细说明)。但RDTSC指令是一条CPU指令,凡是i386平台下Pentium以上的机器均支持,甚至没有平台的限制(我相信i386版本UNIX和Linux下这个方法同样适用,但没有条件试验),而且函数调用的开销是最小的。具有和CPU主频直接对应的速率关系。一个计数相当于1/(CPU主频Hz数)秒,这样只要知道了CPU的主频,可以直接计算出时间。这和QueryPerformanceCount不同,后者需要通过QueryPerformanceFrequency获取当前计数器每秒的计数次数才能换算成时间。这个方法的缺点是:现有的C/C++编译器多数不直接支持使用RDTSC指令,需要用直接嵌入机器码的方式编程,比较麻烦。数据抖动比较厉害。其实对任何计量手段而言,精度和稳定性永远是一对矛盾。如果用低精度的timeGetTime来计时,基本上每次计时的结果都是相同的;而RDTSC指令每次结果都不一样,经常有几百甚至上千的差距。这是这种方法高精度本身固有的矛盾。关于这个方法计时的最大长度,我们可以简单的用下列公式计算:自CPU上电以来的秒数=RDTSC读出的周期数/CPU主频速率(Hz)64位无符号整数所能表达的最大数字是1.8x10^19,在我的Celeron800上可以计时大约700年(书中说可以在200MHz的Pentium上计时117年,这个数字不知道是怎么得出来的,与我的计算有出入)。无论如何,我们大可不必关心溢出的问题。下面是几个小例子,简要比较了三种计时方法的用法与精度//Timer1.cpp使用了RDTSC指令的Timer类//KTimer类的定义可以参见《Windows图形编程》P15〃编译行:CLTimer1.cpp/linkUSER32.lib#include<stdio.h>#include"KTimer.h"main()(unsignedt;KTimertimer;timer.Start();Sleep(1000);t=timer.Stop();printf("Las
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 化工蒸馏工冲突解决强化考核试卷含答案
- 2026年新科教版高中高二生物下册第一单元内环境稳态调节机制卷含答案
- 2026年新科教版初中七年级英语下册第一单元英语动词时态综合卷含答案
- 首饰设计师班组建设评优考核试卷含答案
- 瓦斯防突工安全技能测试知识考核试卷含答案
- 玩具设计师岗后竞赛考核试卷含答案
- 2026年新科教版初中九年级科学下册第三单元天体系统层次结构卷含答案
- 磷酸生产工安全管理测试考核试卷含答案
- 殡葬礼仪师QC管理知识考核试卷含答案
- 纸张整饰工风险评估与管理知识考核试卷含答案
- 2026年江苏南京市高三二模高考政治模拟试卷试题(含答案详解)
- 女生汽车工程专业就业指南
- 人教版三~四年级体育与健康3.5用眼卫生课件(19张)
- 五年级数学小数加减法计算题(简便计算)名师资料
- 统编版历史八年级下学期第13课《中国特色社会主义事业取得新成就》 教学课件
- 2025年四川省雅安市中考数学试卷真题及答案详解(精校打印版)
- 高质量数据集建设和标准化情况介绍
- 2026江苏苏州市常熟市莫城街道(服装城)国有(集体)公司招聘13人备考题库带答案详解(b卷)
- 村委内部控制制度
- 《党政领导干部选拔任用工作条例》知识测试题及答案
- 人教版四年级下册数学第七单元《图形的运动(二)》课件
评论
0/150
提交评论