如何使用SetTimer替换Sleep()?

时间:2016-06-20 14:01:52

标签: c++ winapi sleep

我有一个应用程序需要按顺序逐步执行,如下所示:

<!DOCTYPE html>
<html>
<head>
    <title></title>
    <style type="text/css">
        body
        {
            font-family: Arial;
            font-size: 10pt;
        }
    </style>
    <script src="https://code.jquery.com/jquery-2.2.4.min.js" integrity="sha256-BbhdlvQf/xTY9gja0Dq3HiwQF8LaCRTXxZKRutelT44=" crossorigin="anonymous"></script>
</head>
<body>
    <script src="https://maps.googleapis.com/maps/api/js?sensor=false&libraries=places"></script>
    <script>
        var source, destination;
        var directionsDisplay;
        var directionsService = new google.maps.DirectionsService();
        google.maps.event.addDomListener(window, 'load', function () {
            new google.maps.places.SearchBox($('#txtSource').get(0));
            new google.maps.places.SearchBox($('#txtDestination').get(0));
            directionsDisplay = new google.maps.DirectionsRenderer({ 'draggable': true });
        });

        function GetRoute() {
            var mumbai = new google.maps.LatLng(18.9750, 72.8258);
            var mapOptions = {
                zoom: 7,
                center: mumbai
            };
            map = new google.maps.Map($('#dvMap').get(0), mapOptions);
            directionsDisplay.setMap(map);
            directionsDisplay.setPanel($('#dvPanel').get(0));

            //*********DIRECTIONS AND ROUTE**********************//
            source = $('#txtSource').val();
            destination = $('#txtDestination').val();

            var request = {
                origin: source,
                destination: destination,
                travelMode: google.maps.TravelMode.DRIVING
            };
            directionsService.route(request, function (response, status) {
                if (status == google.maps.DirectionsStatus.OK) {
                    directionsDisplay.setDirections(response);
                }
            });

            //*********DISTANCE AND DURATION**********************//
            var service = new google.maps.DistanceMatrixService();
            service.getDistanceMatrix({
                origins: [source],
                destinations: [destination],
                travelMode: google.maps.TravelMode.DRIVING,
                unitSystem: google.maps.UnitSystem.METRIC,
                avoidHighways: false,
                avoidTolls: false
            }, function (response, status) {
                if (status == google.maps.DistanceMatrixStatus.OK && response.rows[0].elements[0].status != 'ZERO_RESULTS') {
                    var distance = response.rows[0].elements[0].distance.text;
                    var duration = response.rows[0].elements[0].duration.text;
                    var dvDistance = $('#dvDistance');
                    dvDistance.html('');
                    dvDistance.html('Distance: ' + distance + '<br>');
                    dvDistance.html('Duration:' + duration);

                } else {
                    alert('Unable to find the distance via road.');
                }
            });
        }
    </script>
    <table border="0" cellpadding="0" cellspacing="3">
        <tr>
            <td colspan="2">
                Source:
                <input type="text" id="txtSource" value="Bandra, Mumbai, India" style="width: 200px" />
                &nbsp; Destination:
                <input type="text" id="txtDestination" value="Andheri, Mumbai, India" style="width: 200px" />
                <br />
                <input type="button" value="Get Route" onclick="GetRoute()" />
                <hr />
            </td>
        </tr>
        <tr>
            <td colspan="2">
                <div id="dvDistance">
                </div>
            </td>
        </tr>
        <tr>
            <td>
                <div id="dvMap" style="width: 500px; height: 500px">
                </div>
            </td>
            <td>
                <div id="dvPanel" style="width: 500px; height: 500px">
                </div>
            </td>
        </tr>
    </table>
    <br />
</body>
</html>

我们知道,调用Sleep()会导致用户界面“冻结”#34;例如,在尝试移动应用程序窗口时没有响应。

所以,我尝试使用SetTimer(),因为它似乎不会导致UI冻结,但我的问题是如何实现相同的&#34;等待&#34;当使用SetTimer()?

时,它可以用作Sleep()
...
func1();
Sleep(2000);
func2();
Sleep(3000);
func3();
...

感谢您的任何建议。

3 个答案:

答案 0 :(得分:4)

您希望仅在func2()到期后运行ID_Timer2s,因此请在ID_Timer2s到期时运行它。不要马上运行它。

...
func1();
SetTimer(hwnd, ID_Timer2s,2000, Timerfunc2s);
...

VOID CALLBACK Timerfunc2s( 
    HWND hwnd,        
    UINT message,
    UINT idTimer,
    DWORD dwTime)
{ 
    KillTimer(hwnd, ID_Timer2s);
    func2();
    SetTimer(hwnd, ID_Timer3s,3000, Timerfunc3s);
} 

VOID CALLBACK Timerfunc3s( 
    HWND hwnd,        
    UINT message,
    UINT idTimer,
    DWORD dwTime)
{ 
    KillTimer(hwnd, ID_Timer3s);
    func3();
} 

答案 1 :(得分:2)

正如评论中所指出的,这通常不适合GUI应用程序。

但是,它可以适用于一些非常简单的应用程序,例如那些执行不需要用户交互的任务的应用程序,其中GUI不做什么,只提供有关进程的反馈任务。

在这种情况下,并假设代码未在窗口过程的上下文中调用,这样的事情应该有效:

...
func1();
MySleep(hwnd, 2000);
func2();
MySleep(hwnd, 3000);
func3();
...

BOOL MySleepTimerFlag;

void MySleep(HWND hwnd, DWORD timeout)
{
   MySleepTimerFlag = FALSE;
   SetTimer(hwnd, ID_MySleepTimer, timeout, MySleepTimerFunc);
   while (MySleepTimerFlag == FALSE)
   {
      if (!GetMessage(&Msg, NULL, 0, 0)) fail();
      TranslateMessage(&Msg);
      DispatchMessage(&Msg);
   }
}

VOID CALLBACK MySleepTimerFunc( 
    HWND hwnd,        
    UINT message,
    UINT idTimer,
    DWORD dwTime)
{ 
    MySleepTimerFlag = TRUE;
    KillTimer(hwnd, ID_MySleepTimer);
} 

请注意,此代码是从袖口上写下来的;我还没有尝试编译它,更别介意测试它。但它至少应该给你这个想法。

另外,请注意,可以消除全局,如果它让你感到不安 - 将一个指针作为一个窗口属性附加或类似 - 但同样,全局变量在使用时并不是那么糟糕非常简单的应用。 : - )

最后,对MySleep()的调用必须不在窗口过程的上下文中的条件是 important 。这意味着,例如,上面代码的第一部分不能通过选择菜单项来触发。如果是,那么你必须使用Raymond的答案中概述的方法,或者像David建议的那样创建一个新的线程。

...另外,正如Raymond指出的那样,你必须要小心GUI没有任何模态元素,如菜单或对话框,或至少在运行此代码时禁用它们。

答案 2 :(得分:1)

有(至少)3种选择可以做你想要的。第一个,使用计时器,雷蒙德陈覆盖。第二,使用定制的等待功能,哈里约翰斯顿覆盖。第三个选项是创建一个线程,并使用该线程完成工作。

例如:

DWORD WINAPI ThreadProc(LPVOID lpThreadParameter)
{
    func1();
    Sleep(2000);
    func2();
    Sleep(3000);
    func3();
    ...
    return 0;
}

...

DWORD threadId = 0;
HANDLE hThread = CreateThread(NULL, 255, &ThreadProc, NULL, 0, &threadId);

当您需要与UI进行交互时,就会遇到麻烦。您无法从备用线程直接对UI执行操作,但您可以使用的一种技术是post messages到您的UI和/或使用共享状态变量来影响UI线程的行为。< / p>

当然,不言而喻,只要你引入线程,你也会介绍很难诊断错误的可能性,比如竞争条件和死锁。如果func1func2funcN修改全局状态,则需要通过关键部分或其他某种同步方法来同步访问。