Реализация функции Delay

Реализация функции Delay

procedure Delay(dwMilliseconds: Longint);
 var
  iStart, iStop: DWORD;
 begin
  iStart := GetTickCount;
  repeat
  iStop := GetTickCount;
  Application.ProcessMessages;
  until (iStop - iStart) >= dwMilliseconds;
 end;
Взято с сайта: http://www.swissdelphicenter.ch

Примечание от Vit: функция будет "безбожно" жрать процессорное время. Если ожидаемые интервалы задержек достаточно велики, то очень желательно её дополнить следующим образом:

procedure Delay(dwMilliseconds: Longint);

{©Drkb v.3(2007): <a href="http://www.drkb.ru" title="www.drkb.ru">www.drkb.ru</a>,

 ®Vit (Vitaly Nevzorov) - nevzorov@yahoo.com}


 var

  iStart, iStop: DWORD;

 begin

  iStart := GetTickCount;

  repeat

  iStop := GetTickCount;

  sleep(10);

  Application.ProcessMessages;

  until (iStop - iStart) >= dwMilliseconds;

 end;

Команда Sleep будет отдавать время другим приложением, но точность отмеряемого интервала пострадает на 0.01 секунды, что сопоставимо с общей погрешностью предложенного метода.

procedure Delay(msecs: Longint);
 var
  targettime: Longint;
  Msg: TMsg;
 begin
  targettime := GetTickCount + msecs;
  while targettime > GetTickCount do
  if PeekMessage(Msg, 0, 0, 0, PM_REMOVE) then
  begin
  If Msg.message = WM_QUIT Then
  begin
  PostQuitMessage(msg.wparam);
  Break;
  end;
  TranslateMessage(Msg);
  DispatchMessage(Msg);
  end;
 end;
 {
 Note:
 The elapsed time is stored as a DWORD value.
 Therefore, the time will wrap around to zero if the system is
 run continuously for 49.7 days.
}
Взято с сайта: http://www.swissdelphicenter.ch

{
 The Sleep function suspends the execution of the current
 thread for a specified interval.
}

 Sleep(dwMilliseconds: Word);
Взято с сайта: http://www.swissdelphicenter.ch

Примечание от Vit: эта функция страдает многими недостатками, во-первых ни о какой точности речь и не идёт, погрешности среднепотолочные, во-вторых при количестве милисекунд большем 100 функция будет создавать эффект "подвисания" приложения, так как во время её работы приложение не будет отвечать на внешние воздействия. Собственно сама по себе функция sleep вовсе не для отсчёта времени и не для задержки на определённый интервал, её назначение в другом - отдать процессорное время в течение заданного интервала другим процессам. Код можно подправить примерно таким образом:

Procedure Delay (IntervalinSeconds:integer);

var i:integer;

{©Drkb v.3(2007): <a href="http://www.drkb.ru" title="www.drkb.ru">www.drkb.ru</a>,

 ®Vit (Vitaly Nevzorov) - nevzorov@yahoo.com}


begin

 For i:=0 to IntervalinSeconds*100 do

  begin

  sleep(10);

  Application.ProcessMessages;

  end;

end;

Это "поправит" подвисание, но не улучшит точность. Эта функция хороша для задержек на период от секунд до несколько минут когда точность не важна.
Можно так же помудрив сделать и возможность прерывания ожидания по нажатию кнопки, примерно так:
var Canceled:boolean; // должна быть объявлена как глобальная переменная модуля

.....

Function Delay (IntervalinSeconds:integer):boolean; //возвращает true если отработала не прерываясь

{©Drkb v.3(2007): <a href="http://www.drkb.ru" title="www.drkb.ru">www.drkb.ru</a>,

 ®Vit (Vitaly Nevzorov) - nevzorov@yahoo.com}
var i:integer;

begin

 Canceled:=false;

 For i:=0 to IntervalinSeconds*100 do

  begin

  sleep(10);

  Application.ProcessMessages;

  if Canceled then break;

  end;

 Result:=not Canceled;

end;

........

Procedure TForm1.onButton1Click(Sender:TObject);

begin

 Canceled:=true;

end;

Автор: Vit (www.delphist.com, www.drkb.ru, www.unihighlighter.com, www.nevzorov.org)

{
 Including the Sleep in the loop prevents the app from hogging
 100% of the CPU for doing practically nothing but running around the loop.
}

 procedure PauseFunc(delay: DWORD);
 var
  lTicks: DWORD;
 begin
  lTicks := GetTickCount + delay;
  repeat
  Sleep(100);
  Application.ProcessMessages;
  until (lTicks <= GetTickCount) or Application.Terminated;
 end;
Взято с сайта: http://www.swissdelphicenter.ch

procedure Delay(Milliseconds: Integer);
 {by Hagen Reddmann}
 var
  Tick: DWord;
  Event: THandle;
 begin
  Event := CreateEvent(nil, False, False, nil);
  try
  Tick := GetTickCount + DWord(Milliseconds);
  while (Milliseconds > 0) and
  (MsgWaitForMultipleObjects(1, Event, False, Milliseconds, QS_ALLINPUT) <> WAIT_TIMEOUT) do
  begin
  Application.ProcessMessages;
  Milliseconds := Tick - GetTickcount;
  end;
  finally
  CloseHandle(Event);
  end;
 end;
Взято с сайта: http://www.swissdelphicenter.ch

Отправить комментарий

Проверка
Антиспам проверка
Image CAPTCHA
...