C++中如何精确计时

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C++中如何精确计时相关的知识,希望对你有一定的参考价值。

console程序里如何每隔一段时间调用某个函数,请高手赐教.

下面是我以前找的些资料。
1.使用CTime类
CString str;
//获取系统时间
CTime tm;
tm=CTime::GetCurrentTime();
str=tm.Format("现在时间是%Y年%m月%d日 %X");
MessageBox(str,NULL,MB_OK);
2: 得到系统时间日期(使用GetLocalTime)
SYSTEMTIME st;
CString strDate,strTime;
GetLocalTime(&st);
strDate.Format("%4d-%2d-%2d",st.wYear,st.wMonth,st.wDay);
strTime.Format("%2d:%2d:%2d",st.wHour,st.wMinute,st.wSecond);
AfxMessageBox(strDate);
AfxMessageBox(strTime);
3.使用GetTickCount
//获取程序运行时间
long t1=GetTickCount();//程序段开始前取得系统运行时间(ms)
Sleep(500);
long t2=GetTickCount();();//程序段结束后取得系统运行时间(ms)
str.Format("time:%dms",t2-t1);//前后之差即 程序运行时间
AfxMessageBox(str);
//获取系统运行时间
long t=GetTickCount();
CString str,str1;
str1.Format("系统已运行 %d时",t/3600000);
str=str1;
t%=3600000;
str1.Format("%d分",t/60000);
str+=str1;
t%=60000;
str1.Format("%d秒",t/1000);
str+=str1;
AfxMessageBox(str);

用VC++获取系统时间几种方法
用VC++获取系统时间几种方法
Q:如何获取时间?精度如何?

A:
1 使用time_t time( time_t * timer ) 精确到秒
  计算时间差使用double difftime( time_t timer1, time_t timer0 )

2 使用clock_t clock() 得到的是CPU时间 精确到1/CLOCKS_PER_SEC秒

3 使用DWORD GetTickCount() 得到的是系统运行的时间 精确到毫秒

4 如果使用MFC的CTime类,可以用CTime::GetCurrentTime() 精确到秒

5 要获取高精度时间,可以使用
BOOL QueryPerformanceFrequency(LARGE_INTEGER *lpFrequency)获取系统的计数器的频率
BOOL QueryPerformanceCounter(LARGE_INTEGER *lpPerformanceCount)获取计数器的值
然后用两次计数器的差除以Frequency就得到时间。

6 还有David的文章中提到的方法:
Multimedia Timer Functions
The following functions are used with multimedia timers.
timeBeginPeriod/timeEndPeriod/timeGetDevCaps/timeGetSystemTime
timeGetTime/timeKillEvent/TimeProc/timeSetEvent 精度很高

Q:GetTickCount()函数,说是毫秒记数,是真的吗,还是精确到55毫秒?

A:
GetTickCount()和GetCurrentTime()都只精确到55ms(1个tick就是55ms)。如果要精确到毫秒,应该使用timeGetTime函数或QueryPerformanceCounter函数。具体例子可以参考QA001022 "VC++中使用高精度定时器"、QA001813 "如何在Windows实现准确的定时"和QA004842 "timeGetTime函数延时不准"。

Q:vc++怎样获取系统时间,返回值是什么类型的变量呢?

GetSystemTime返回的是格林威志标准时间
GetLocalTime,和上面用法一样,返回的是你所在地区的时间,中国返回的是北京时间

VOID GetSystemTime(
LPSYSTEMTIME lpSystemTime // address of system time structure
);
函数就可以获得了,其中LPSYSTEMTIME 是个结构体
含:年,月,日,周几,小时,分,秒,毫秒。

LARGE_INTEGER litmp;
LONGLONG QPart1,QPart2;
double dfMinus, dfFreq, dfTim;
QueryPerformanceFrequency(&litmp);
dfFreq = (double)litmp.QuadPart;// 获得计数器的时钟频率
QueryPerformanceCounter(&litmp);
QPart1 = litmp.QuadPart;// 获得初始值
do

QueryPerformanceCounter(&litmp);
QPart2 = litmp.QuadPart;//获得中止值
dfMinus = (double)(QPart2-QPart1);
dfTim = dfMinus / dfFreq;// 获得对应的时间值,单位为秒
while(dfTim<0.001);
参考技术A 1.SetTimer 创建新的线程执行函数
2.在循环中取得系统时钟进行条件判断就可以了:localtime()
3.用多线程可以实现

如何在 C++ 中使用计时器在给定时间内强制输入?

【中文标题】如何在 C++ 中使用计时器在给定时间内强制输入?【英文标题】:How to use a timer in C++ to force input within a given time? 【发布时间】:2015-05-10 18:49:25 【问题描述】:

我想在 C++ 中实现超时功能。

如果用户在 2 秒内没有输入值,那么程序必须显示超时语句并再次询问输入

EX(输出屏幕):

Timer=0;  
Please enter the input:       //if input is not given within 2 seconds then  
Time-out: 2 seconds  
Timer again set to 0  
Please enter the input:  //if input is not given within 2 seconds then  
Time-out: 2 seconds  
Timer again set to 0  
Please enter the input:22  
Data accepted  
Terminate the program`

代码:

#include<iostream>  
 #include<time.h>  
 using namespace std;  
 int main()  
  
    clock_t endwait;  
    endwait =  2000 ;  
   cout<<endwait;  
   while (clock() < endwait)  
    
         cout<<"Please enter the input:";  
    
   return 0;  
 

我已经处理了上面的代码。但这仅在进入 WHILE 循环时发生。我该怎么做才能得到所需的输出。

【问题讨论】:

没有标准的干净方式只能在有限的时间内接受输入, 你必须让函数获取输入一个线程并且可以使用wait_for @NathanOliver 你能详细说明一下吗? 平台特定,等待按键事件。你用的是什么平台? 带代码块的 Windows 7 【参考方案1】:

恐怕以标准方式不可能,因为默认情况下,I/O 操作会阻塞调用进程,直到它完成或遇到错误。 您可以通过创建一个线程来解决它,该线程检查输入是否已完成并在需要时进入睡眠状态。但这并不实际。

问题在于iostream/FILEs 给你的抽象:你无权访问底层源,操作系统也“理解”,因此能够为你提供那种功能(即 I/O 轮询)。

【讨论】:

【参考方案2】:

单独的线程是不够的,因为超时后控制台读取函数还在运行。

在 POSIX 上,您可以设置一个计时器,该计时器产生一个信号并导致读取失败并显示 -EINTR

在 Windows 上,您可以使用 ReadConsoleInputWaitForSingleObject... 实现低级控制台 I/O 超时...但是您需要自己进行行缓冲。

另一个想法是在OVERLAPPED 模式下使用ReadFile 并等待超时完成事件,但这不适用于控制台,请参阅Using overlapped IO for console input?

最后,新的 Windows 版本(Vista 和更高版本)允许您使用 CancelIoEx 克服“阻塞读取不会被超时取消”问题。如果你从一个单独的线程调用它,它将触发ReadFile 提前返回;您不必自己实现行缓冲。

【讨论】:

【参考方案3】:
#include <iostream>
#include <thread>
#include <chrono>
#include <mutex>
#include <condition_variable>
using namespace std;
condition_variable cv;

int value;

void read_value() 
    cin >> value;
    cv.notify_one();


int main()

    cout << "Please enter the input: ";
    thread th(read_value);

    mutex mtx;
    unique_lock<mutex> lck(mtx);
    while (cv.wait_for(lck, chrono::seconds(2)) == cv_status::timeout)
    
        cout << "\nTime-Out: 2 second:";
        cout << "\nPlease enter the input:";
    
    cout << "You entered: " << value << '\n';

    th.join();

    return 0;

输出:

Please enter the input:  
Time-Out: 2 second:  
Please enter the input:  
Time-Out: 2 second:  
Please enter the input:22  
You entered: 22  

【讨论】:

虽然这看起来可行,但实际上并没有使 cin 超时。例如,如果用户要在提示之后键入一个值,但在超时之前没有按 Enter,但在随后的提示中,用户确实及时按了 Enter。用户输入的所有值,无论他们是否按回车键,都将附加到您的值,因为流永远不会刷新。您可以考虑在某个时候停止并重新启动您的流或刷新输入流。很好地使用了 condition_variable!【参考方案4】:

我认为没有必要为此使用复杂的代码(多线程或互斥锁)。请看下面的代码:

#include <iostream>
#include <time.h>
#include <conio.h>

using namespace std;

int main()

    int numInput;

    clock_t start = clock();

    cout << "Timer: 2 sec"<<endl;

    cout << "Please enter the input: ";

    while ( ! _kbhit() ) //Check for keyboard hit
    
        //Check if 2 sec timer expired or not
        if (((clock () - start)/ CLOCKS_PER_SEC ) >= 2) 
        
            cout << "\nTimeout  2 sec . . ." << endl;
            cout << "Please enter the input: ";
            start = clock();                            //Set new timer
        
    

    //Get the input here
    cin >> numInput;

    cout << "Data accepted: " << numInput << endl;

    _getch();

    return 0;

【讨论】:

【参考方案5】:

我已经使用 kbhit() 函数来解决您的问题。代码如下:-

    #include <conio.h>
    #include <iostream>
    #include <windows.h>

    using namespace std;

    int main()
    
        int i;
        DWORD start_time, check_time;
        start_time=GetTickCount();
        check_time=start_time+2000;
        while(!kbhit())
        
            cout<<"Please enter the input:"<<endl;
            while((check_time>GetTickCount()))
            
                if (kbhit())
                
                    i=getch();
                    cout<<"Data accepted"<<endl;
                    return 0;
                
            
            cout<<"Time-out: 2 seconds"<<endl;
            cout<<"Timer again set to 0"<<endl;
            start_time=GetTickCount();
            check_time=start_time+2000;
        
    return 0;

    

【讨论】:

以上是关于C++中如何精确计时的主要内容,如果未能解决你的问题,请参考以下文章

怎么用C++做日期相减(要求精确到分钟)

如何在VB中做精确度高的计时器??

我可以精确计时重叠的 ReadFileEx 操作吗?

在Java中如何实现较为精确的定时器

.NET 中的精确计时

c++中如何精确输出的位数?