Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Sonstige Fragen zu Delphi (https://www.delphipraxis.net/19-sonstige-fragen-zu-delphi/)
-   -   Delphi ScheduledFuture in Delphi? (https://www.delphipraxis.net/141417-scheduledfuture-delphi.html)

WorstNightmare 8. Okt 2009 15:29


ScheduledFuture in Delphi?
 
Hallo,

in Java gibt es ja die Klasse ScheduledFuture, um Befehle nach einer gewissen Zeitspanne auszuführen. In Delphi könnte man ja einen TTimer mit dem entsprechenden Intervall benuzten und diesen in der OnTimer Funktion wieder abschalten, allerdings funktionieren die ja nicht in Threads. Gibt es Timer-Komponenten die auch in Threads gehen oder gar eine Klasse wie ScheduledFuture in Delphi?

Dax 8. Okt 2009 15:37

Re: ScheduledFuture in Delphi?
 
In Threads kannst du statt dem Timer doch einfach Sleep(dword) benutzen?

WorstNightmare 8. Okt 2009 15:49

Re: ScheduledFuture in Delphi?
 
Vielleicht sollte ich noch dazu erwähnen, dass es sich um Indy-Server Threads handelt. Ein Sleep hätte daher fatale Folgen, da dann auch die Pakete nicht mehr verabeitet werden würden, die währenddessen reinkommen.

himitsu 8. Okt 2009 15:52

Re: ScheduledFuture in Delphi?
 
erstell dir doch einfach einen weiteren Thread, welchen du als ScheduleQueue benutzt

oder nimm einem Timer im Hauptthread (auf der Form), welcher dann entsprechend nur die Event-Starts an die Threads weiterreicht.

WorstNightmare 8. Okt 2009 16:46

Re: ScheduledFuture in Delphi?
 
Zitat:

erstell dir doch einfach einen weiteren Thread, welchen du als ScheduleQueue benutzt
Kannst du das vielleicht noch etwas mehr erläutern? :?
Wie soll ich denn prüfen ob die Zeit um ist? Einfach GetTickCount mit dem Startzeitpunkt vergleichen? Ist das denn genau?

Deine letztere Möglichkeit wäre etwas umständlich...

Deep-Sea 12. Okt 2009 08:39

Re: ScheduledFuture in Delphi?
 
Zitat:

Zitat von WorstNightmare
Zitat:

erstell dir doch einfach einen weiteren Thread, welchen du als ScheduleQueue benutzt
Kannst du das vielleicht noch etwas mehr erläutern? :?
Wie soll ich denn prüfen ob die Zeit um ist? Einfach GetTickCount mit dem Startzeitpunkt vergleichen? Ist das denn genau?

Wenn du einen Thread nur dafür nutzt, dann kannst du ya wieder mit Sleep arbeiten. Aber Sleep wird kaum genauer sein als GetTickCount oder ein Timer *glaub*. Gegenfrage: Wie genau muss es denn sein?


Zitat:

Zitat von WorstNightmare
Deine letztere Möglichkeit wäre etwas umständlich...

Wieso denn nicht? Auf den ersten Blick scheint mir dies die einfachste Möglichkeit (aber wohl nicht die schönste). :-D

Luckie 12. Okt 2009 08:43

Re: ScheduledFuture in Delphi?
 
Zitat:

Zitat von WorstNightmare
Gegenfrage: Wie genau muss es denn sein?

Wenn man schon nach der Genauigkeit fragen muss, ist es schon der falsche Ansatz. Windows ist kein Echtzeitbetriebssystem.

DataCool 12. Okt 2009 11:56

Re: ScheduledFuture in Delphi?
 
Zitat:

Vielleicht sollte ich noch dazu erwähnen, dass es sich um Indy-Server Threads handelt.
Hi,

kannst bitte kurz erklären was Du genau machen möchtest ?!
Ich kann mir nämlich beim besten Willen nicht vorstellen warum Du innerhalb eines Indy-Server-Threads eine gewissen Zeitspanne warten möchtest ?!

Wenn Du außerhalb des "Indy-Server-Threads" etwas mit ein paar Clients/oder allen machen möchtest, z.B. alle xyz Sekunden ein Kommando schicken,
dann solltest Du dafür einen Extra-Timer oder noch besser Extra Thread verwenden.

Greetz Data

WorstNightmare 17. Okt 2009 18:19

Re: ScheduledFuture in Delphi?
 
So,

ich bin aus dem Urlaub zurück und habe wieder angefangen mich mit dem Projekt zu beschäftigen.

Es handelt sich um einen Server zu einem MMORPG und der Server muss eben auch die Buffs, die der Spieler gerade aktiviert hat verwalten. Diese deaktivieren sich immer nach einer Zeit, manche nach 4s und manche erst nach 15min. Daher brauche ich eben sowas wie Timer, welche dem Client mitteilen, die entsprechenden Buffs nach der richtigen Zeit abzuschalten, es muss nicht 100% genau sein, es sollten nur nicht mehr als 500ms Abweichung sein.

Ich dachte mir also, ich bastele mir einen Scheduler-Thread, der die ganze Zeit läuft. Dieser würde dann ein Dictionary führen, dessen Key der TickCount in der Zukunft ist und die Value der auszuführende Job (als procedure). In der Execute Schleife wird dann einmal der aktuelle TickCount ermittelt und dann mit allen Keys im Dictionary verglichen und die, welche passen werden ausgeführt.

Folgende Bedenken habe ich:
Zu Bestzeiten könnten im Dictionary 600 Einträge sein, die in jedem Execute verglichen werden, könnte es da irgendwelche Performanceprobleme geben? Sollte ich vielleicht mehrere Scheduler laufen lassen, die jeweils 100 Einträge verwalten oder würde das keinen Sinn machen?
Zusätzlich laufen dann auch noch bis zu 20 Indy-Listener Threads und bis zu 300 Client-Verbindungsthreads, falls das mitbeachtet werden muss.

Was sagt ihr dazu?

thkerkmann 17. Okt 2009 18:43

Re: ScheduledFuture in Delphi?
 
Sortier dein Dictionary nach Ablaufzeit, und du brauchst nur immer die ersten anzuschauen, bis einer kommt der noch warten muss.

Apollonius 17. Okt 2009 19:17

Re: ScheduledFuture in Delphi?
 
Verwende den Windows Thread Pool: MSDN-Library durchsuchenCreateTimerQueueTimer

WorstNightmare 17. Okt 2009 19:46

Re: ScheduledFuture in Delphi?
 
Zitat:

Zitat von Apollonius
Verwende den Windows Thread Pool: MSDN-Library durchsuchenCreateTimerQueueTimer

Das sieht doch mal sauber aus. Ich habe mich mal dran versucht, allerdings wird OnTimer irgendwie nicht aufgerufen, stattdessen stürzt das Programm mit einem APPCRASH ab (keine Rückmeldung) :(

Delphi-Quellcode:
unit Scheduler;

interface

uses Windows;

type
  TSchedule = reference to procedure;

  TScheduler = class
  private
    FQueue: THandle;
  public
    constructor Create;
    destructor Destroy; override;

    procedure AddSchedule(Milliseconds: Cardinal; Proc: TSchedule);
  end;

implementation

{ TScheduler }

constructor TScheduler.Create;
begin
  FQueue := CreateTimerQueue;
end;

destructor TScheduler.Destroy;
begin
  DeleteTimerQueue(FQueue);

  inherited;
end;

procedure OnTimer(Context: Pointer; Success: Boolean); stdcall;
var
  Proc: TSchedule;
begin
  Proc := TSchedule(Context);
  Proc;
end;

procedure TScheduler.AddSchedule(Milliseconds: Cardinal; Proc: TSchedule);
var
  Timer: THandle;
begin
  CreateTimerQueueTimer(Timer, FQueue, OnTimer, @Proc, Milliseconds, 0, WT_EXECUTEONLYONCE);
end;

end.
Aufruf:
Delphi-Quellcode:
var
  S: TScheduler;
begin
  S := TScheduler.Create;
  S.AddSchedule(5000, procedure begin ShowMessage('test') end);
end;
Weiß jemand was falsch ist?

Edit: Es liegt am reference to proceudre, mache ich das ShowMessage direkt in OnTimer geht es.

Apollonius 17. Okt 2009 19:55

Re: ScheduledFuture in Delphi?
 
Die Übergabe des Kontext-Parameter wird mit dem Pointer nicht funktionieren, da Proc auf dem Stack liegt. Ich kenne mich leider mit den Interna von den neuen Routinenreferenzen nicht aus, meine mich jedoch zu erinnern, dass es sich um Interfaces handelt. In diesem Fall müsstest du Pointer(Proc) übergeben und zusätzlich noch auf die Referenzzählung achten. In jedem Fall auf der sicheren Seite bist du, wenn du mit New einen PSchedule allozierst, ihn mit Proc befüllst und dem Thread übergibst, welcher ihn am Ende freigibt.

WorstNightmare 17. Okt 2009 20:16

Re: ScheduledFuture in Delphi?
 
Zitat:

Ich kenne mich leider mit den Interna von den neuen Routinenreferenzen nicht aus
Hatte das eh nur aus Gemütlichkeit gemacht, wollte es dann in procedure of object umwandeln.

Hab es jetzt so:

Delphi-Quellcode:
unit Scheduler;

interface

uses Windows, SysUtils;

type
  TSchedule = procedure of object;

  TDescriptor = record
    Proc: TSchedule;
    Queue: THandle;
    Timer: PHandle;
  end;
  PDescriptor = ^TDescriptor;

  TScheduler = class
  private
    FQueue: THandle;
  public
    constructor Create;
    destructor Destroy; override;

    procedure AddSchedule(Milliseconds: Cardinal; Proc: TSchedule);
  end;

implementation

{ TScheduler }

constructor TScheduler.Create;
begin
  FQueue := CreateTimerQueue;
end;

destructor TScheduler.Destroy;
begin
  DeleteTimerQueue(FQueue);

  inherited;
end;

procedure OnTimer(Context: Pointer; Success: Boolean); stdcall;
begin
  PDescriptor(Context)^.Proc;

  DeleteTimerQueueTimer(PDescriptor(Context)^.Queue, Cardinal(PDescriptor(Context)^.Timer^), INVALID_HANDLE_VALUE);

  Dispose(Context);
end;

procedure TScheduler.AddSchedule(Milliseconds: Cardinal; Proc: TSchedule);
var
  Timer: THandle;
  PDesc: PDescriptor;
begin
  New(PDesc);
  PDesc^.Proc := Proc;
  PDesc^.Queue := FQueue;
  PDesc^.Timer := @Timer;

  if not CreateTimerQueueTimer(Timer, FQueue, OnTimer, PDesc, Milliseconds, 0, WT_EXECUTEONLYONCE) then
    raise Exception.Create('Creating a timer failed!');
end;

end.
Die Funktion des Schedulers an sich funktioniert jetzt, die Message erscheint. Allerdings wollte ich danach gerne auch aufräumen und DeleteTimerQueueTimer aufrufen, dafür brauche ich aber ein paar Parameter die ich irgendwie noch da rein kriegen musste :angel2:
Und da geht wohl was schief, ich hab's mit Pointern nicht so besonders :|

Apollonius 17. Okt 2009 20:20

Re: ScheduledFuture in Delphi?
 
Es gibt keinen Grund, einen Zeiger auf das Timer-Handle in TDescriptor zu speichern. Nimm das Handle selbst.
Der letzte Parameter von DeleteTimerQueueTimer sollte wahrscheinlich 0 sein, INVALID_HANDLE_VALUE blockiert. Auf der anderen Seite weiß ich nicht, ob man aus dem Callback überhaupt seinen eigenen Timer löschen kann.

WorstNightmare 17. Okt 2009 20:35

Re: ScheduledFuture in Delphi?
 
Zitat:

Zitat von Apollonius
Es gibt keinen Grund, einen Zeiger auf das Timer-Handle in TDescriptor zu speichern. Nimm das Handle selbst.
Der letzte Parameter von DeleteTimerQueueTimer sollte wahrscheinlich 0 sein, INVALID_HANDLE_VALUE blockiert. Auf der anderen Seite weiß ich nicht, ob man aus dem Callback überhaupt seinen eigenen Timer löschen kann.

Das hab ich mir auch schon gedacht, ich weiß nur nicht wann ich es sonst machen soll :(

Delphi-Quellcode:
procedure OnTimer(Context: Pointer; Success: Boolean); stdcall;
begin
  PDescriptor(Context)^.Proc;

  try
    if not DeleteTimerQueueTimer(PDescriptor(Context)^.Queue, PDescriptor(Context)^.Timer, 0) then
      raise Exception.Create('Deleting a timer failed!');
  finally
    Dispose(Context);
  end;
end;

procedure TScheduler.AddSchedule(Milliseconds: Cardinal; Proc: TSchedule);
var
  Timer: THandle;
  PDesc: PDescriptor;
begin
  New(PDesc);
  PDesc^.Proc := Proc;
  PDesc^.Queue := FQueue;

  if not CreateTimerQueueTimer(Timer, FQueue, OnTimer, PDesc, Milliseconds, 0, WT_EXECUTEONLYONCE) then
    raise Exception.Create('Creating a timer failed!');

  PDesc^.Timer := Timer;
end;
"Deleting a timer failed!" tritt auf.

Apollonius 17. Okt 2009 20:43

Re: ScheduledFuture in Delphi?
 
Igitt, diese Race Condition mit Timer = 0 habe ich glatt übersehen. Sie sollte sich allerdings durch Pollen in OnTimer beheben lassen. Was sagt eigentlich GetLastError?

WorstNightmare 17. Okt 2009 20:53

Re: ScheduledFuture in Delphi?
 
Hab es letztendlich so gelöst:

Delphi-Quellcode:
unit Scheduler;

interface

uses Windows, SysUtils, Generics.Collections;

type
  TSchedule = procedure of object;
  TScheduler = class;

  TDescriptor = record
    Proc: TSchedule;
    Timer: THandle;
    Sched: TScheduler;
  end;
  PDescriptor = ^TDescriptor;

  TScheduler = class
  private
    FQueue: THandle;
    FRems: TList<THandle>;

    procedure RemoveInactiveTimers;
  public
    constructor Create;
    destructor Destroy; override;

    procedure AddSchedule(Milliseconds: Cardinal; Proc: TSchedule);

    procedure SetTimerRemovable(Timer: THandle);
  end;

implementation

{ TScheduler }

constructor TScheduler.Create;
begin
  FQueue := CreateTimerQueue;
  FRems := TList<THandle>.Create;
end;

destructor TScheduler.Destroy;
begin
  RemoveInactiveTimers;
  DeleteTimerQueue(FQueue);
  FRems.Free;

  inherited;
end;

procedure OnTimer(Context: Pointer; Success: Boolean); stdcall;
begin
  try
    PDescriptor(Context)^.Proc;
    PDescriptor(Context)^.Sched.SetTimerRemovable(PDescriptor(Context)^.Timer);
  finally
    Dispose(Context);
  end;
end;

procedure TScheduler.SetTimerRemovable(Timer: THandle);
begin
  FRems.Add(Timer);
end;

procedure TScheduler.AddSchedule(Milliseconds: Cardinal; Proc: TSchedule);
var
  Timer: THandle;
  PDesc: PDescriptor;
begin
  RemoveInactiveTimers;

  New(PDesc);
  PDesc^.Proc := Proc;
  PDesc^.Sched := Self;

  if not CreateTimerQueueTimer(Timer, FQueue, OnTimer, PDesc, Milliseconds, 0, WT_EXECUTEONLYONCE) then
    raise Exception.Create('Creating a timer failed!');

  PDesc^.Timer := Timer;
end;

procedure TScheduler.RemoveInactiveTimers;
var
  i: Integer;
begin
  for i := 0 to FRems.Count - 1 do
    if not DeleteTimerQueueTimer(FQueue, FRems[i], 0) then
      raise Exception.Create('Deleting a timer failed!');

  FRems.Clear;
end;

end.
Ich denke, den Timer während OnTimer ausgeführt wird zu terminieren wird einfach nicht möglich sein. So ist es nicht soo toll, beendet aber zumindestens die meisten der fertigen Timer-Threads.
Vielleicht muss ich noch irgendwo eine CriticalSection wegen dem RemoveInactive einführen (weil auf die Liste theoretisch ja gleichzeitig zugegriffen werden kann), aber ansonsten ist das Teil ziemlich fertig :)

Danke für die Hilfe! :dp:


Alle Zeitangaben in WEZ +1. Es ist jetzt 12:43 Uhr.

Powered by vBulletin® Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz