Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Object-Pascal / Delphi-Language (https://www.delphipraxis.net/32-object-pascal-delphi-language/)
-   -   Delphi Events und Threads in Schleife erzeugen (https://www.delphipraxis.net/62904-events-und-threads-schleife-erzeugen.html)

quakergod 11. Feb 2006 07:04


Events und Threads in Schleife erzeugen
 
Hallo, ich habe folgenden Code um Threads & Events zu erzeugen und sie mit Daten zu füttern:

Delphi-Quellcode:
procedure TMainform.ButtonClick(Sender: TObject);
var dw: DWORD;
  ev1, ev2: TEvent;
  t1,t2: TevtThread;    
  events: array[0..1] of THandle;
  FStatus: array[0..3] of string;
  i: integer;
begin
  FDLAbort := False;
  button.Enabled := False;

  ev1 := nil;
  ev2 := nil;
  try
    FDLThreads := 2;

         // Events anlegen
    ev1 := TEvent.Create(nil, False, False, '');
    ev2 := TEvent.Create(nil, False, False, '');
         // Threads anlegen
    t1 := TEvtThread.Create(ev1);
    t2 := TEvtThread.Create(ev2);
         // Downloadschleife
    events[0] := ev1.Handle;
    events[1] := ev2.Handle;

    while FDLThreads > 0 do begin
         // warten auf Thread oder Input
      dw := MsgWaitForMultipleObjects(2, events, False, WF_TIMEOUT {INFINITE}, QS_ALLINPUT);
      case dw of
        WAIT_OBJECT_0 + 0: SetNextDLFile(t1, FStatus[0]);
        WAIT_OBJECT_0 + 1: SetNextDLFile(t2, FStatus[1]);
        WAIT_OBJECT_0 + Length(events): HandleDLUI;
        WAIT_TIMEOUT: HandleDlUI;
      else
          // Fehler
        Break;
      end;
    end;
  finally
    button.Enabled := True;
  end;
end;
der code funktioniert auch ohne Probleme, jedoch möchte ich die anzahl der threads variabel halten,
deswegen hab ich den code so umgeschrieben:

Delphi-Quellcode:
type
  TgmDownload = record
    Event: array[0..20] of TEvent;
    EventHandle: array[0..20] of THandle;
    Thread: array[0..20] of TevtThread;
  end;

var
  FgmDownload: TgmDownload;


procedure TMainform.Button2Click(Sender: TObject);
var dw: DWORD;
  FStatus: array[0..3] of string;
  i: integer;
begin
  FDLAbort := False;
  button2.Enabled := False;

  FDLThreads := strtoint(edtDLThreads.Text);
  try
    for i := 1 to FDLThreads - 1 do
    begin
      FgmDownload.Event[i] := nil;
      FgmDownload.Event[i] := TEvent.Create(nil, False, False, '');
      FgmDownload.Thread[i] := TEvtThread.Create(FgmDownload.Event[i]);
      FgmDownload.EventHandle[i] := FgmDownload.Event[i].Handle;
    end;
   
    while FDLThreads > 0 do begin
         // warten auf Thread oder Input
      dw := MsgWaitForMultipleObjects(2, FgmDownload.EventHandle, False, WF_TIMEOUT {INFINITE}, QS_ALLINPUT);
      case dw of
        WAIT_OBJECT_0 + 0: SetNextDLFile(FgmDownload.Thread[0], FStatus[0]);
        WAIT_OBJECT_0 + 1: SetNextDLFile(FgmDownload.Thread[1], FStatus[0]);
        WAIT_OBJECT_0 + 2: SetNextDLFile(FgmDownload.Thread[2], FStatus[0]);
        WAIT_OBJECT_0 + 3: SetNextDLFile(FgmDownload.Thread[3], FStatus[0]);
        WAIT_OBJECT_0 + 4: SetNextDLFile(FgmDownload.Thread[4], FStatus[0]);
        WAIT_OBJECT_0 + 5: SetNextDLFile(FgmDownload.Thread[5], FStatus[0]);
        WAIT_OBJECT_0 + 6: SetNextDLFile(FgmDownload.Thread[6], FStatus[0]);
        WAIT_OBJECT_0 + 7: SetNextDLFile(FgmDownload.Thread[7], FStatus[0]);
        WAIT_OBJECT_0 + 8: SetNextDLFile(FgmDownload.Thread[8], FStatus[0]);
        WAIT_OBJECT_0 + 9: SetNextDLFile(FgmDownload.Thread[9], FStatus[0]);
        WAIT_OBJECT_0 + Length(FgmDownload.EventHandle): HandleDLUI;
        WAIT_TIMEOUT: HandleDlUI;
      else
         // Fehler
        Break;
      end;
    end;
  finally
    button2.Enabled := True;
  end;
end;
ich sehe leider keinen fehler, aber es kommt nach ein par sekunden zu einer nichtssagende Speicherexception ?!

Weiss jemand was ich falsch gemacht habe oder wo der Fehler liegt? (weiss nicht mehr weiter)

Gruß

gsh 11. Feb 2006 08:37

Re: Events und Threads in Schleife erzeugen
 
Um meine bescheidene Meinung abzugeben ich würd des anderes machen:
und zwar nicht des im Record alles arrays sind sondern des die Var "FgmDownload: TgmDownload;" ein Array of Record ist:
Delphi-Quellcode:
type
  TgmDownload = record
    Event: TEvent;
    EventHandle: THandle;
    Thread: TevtThread;
  end;

var
  FgmDownload: array of TgmDownload; //Außerdem wenn die länge variable sein soll dann mach keine Konstanten rein und mach ein Dynamisches Array


procedure TMainform.Button2Click(Sender: TObject);
var dw: DWORD;
  FStatus: array[0..3] of string;
  i: integer;
begin
  FDLAbort := False;
  button2.Enabled := False;

  FDLThreads := strtoint(edtDLThreads.Text);
  SetLength(FgmDownload, FDLThreads) //Hier wird dann die Länge vom dyn. Array gesetzt
  try
    for i := 0 {Der 1 kann ein Fehler sein, weil des array bei null beginnt} to FDLThreads - 1 do
    begin
      FgmDownload[i].Event := nil; //Ich glaub die kann man weg lassen
      FgmDownload[i].Event := TEvent.Create(nil, False, False, ''); //wird hier nämlich sowieso überschrieben
      FgmDownload[i].Thread := TEvtThread.Create(FgmDownload[i].Event);
      FgmDownload[i].EventHandle := FgmDownload[i].Event.Handle;
    end;
   
    while FDLThreads > 0 do begin //Eine Integer Variable sehe aber nirgenwo wo sie kleiner wird (endlosschleife?)
         // warten auf Thread oder Input
      dw := MsgWaitForMultipleObjects(2, FgmDownload.EventHandle, False, WF_TIMEOUT {INFINITE}, QS_ALLINPUT);
      case dw of
        WAIT_OBJECT_0 + 0: SetNextDLFile(FgmDownload.Thread[0], FStatus[0]); //Keine Ahnung
        WAIT_OBJECT_0 + 1: SetNextDLFile(FgmDownload.Thread[1], FStatus[0]);
        WAIT_OBJECT_0 + 2: SetNextDLFile(FgmDownload.Thread[2], FStatus[0]);
        WAIT_OBJECT_0 + 3: SetNextDLFile(FgmDownload.Thread[3], FStatus[0]);
        WAIT_OBJECT_0 + 4: SetNextDLFile(FgmDownload.Thread[4], FStatus[0]);
        WAIT_OBJECT_0 + 5: SetNextDLFile(FgmDownload.Thread[5], FStatus[0]);
        WAIT_OBJECT_0 + 6: SetNextDLFile(FgmDownload.Thread[6], FStatus[0]);
        WAIT_OBJECT_0 + 7: SetNextDLFile(FgmDownload.Thread[7], FStatus[0]);
        WAIT_OBJECT_0 + 8: SetNextDLFile(FgmDownload.Thread[8], FStatus[0]);
        WAIT_OBJECT_0 + 9: SetNextDLFile(FgmDownload.Thread[9], FStatus[0]);
        WAIT_OBJECT_0 + {Length(}FgmDownload[?].EventHandle{)}: HandleDLUI; //Length von einem Handle???? Des ist schon ne zahl, AUßDERDEM: schlimmer fehler es ist der Array Eintrag (z.b. [1]) nicht angeben d.h. er will des ganze Array ding nehmen und nicht den wert daher kommt wahrscheinlich der fehler
        WAIT_TIMEOUT: HandleDlUI;
      else
         // Fehler
        Break;
      end;
    end;
  finally
    button2.Enabled := True;
  end;
end;
Hoffe ich konnte dir helfen (Ungetestet, es kann sein des ich irgendwo was vergessen hab)


//edit: Einen Satz verbessert

quakergod 12. Feb 2006 19:09

Re: Events und Threads in Schleife erzeugen
 
so, ich weiss nun wo der fehler liegt (danke gsh!)
kann ihn aber leider nicht beheben! :(

ich hab den Eintrag
Delphi-Quellcode:
EventHandle: THandle;
aus dem record genommen und folgendes in die buttonclick prozedur eingefügt:
Delphi-Quellcode:
var EventList: array[0..1] of THandle;
wenn ich nun 2 Threads mache wird einer gemacht, bei einem wird nichts gemacht, ein problem ist wenn ich
Delphi-Quellcode:
WAIT_OBJECT_0 + Length(EventList): HandleDLUI;
habe kann ich beim array EventList nicht dynamisch die Länge festlegen "weil ein Konstantenausdruck erforderlich" ist :gruebel:

weiss jemand rat? :roll:

edit, hat jemand schonmal mit Threads + Events und dynamischer Threadanzahl gearbeitet und könnte mir den Code schicken / posten? :angel2:

gsh 12. Feb 2006 20:05

Re: Events und Threads in Schleife erzeugen
 
poste bitte mal wieder deinen gesamten Code. Kann mir deine Änderunge gerade irgenwie nicht vorstellen.

ich würd dir ja gern etwas code von meinen Programmen geben es ist nur so des ich des ziemlich anderes Programmiere. Außerdem hab ich mit Events noch nie wirklich was getan (mit Threads schon, glaube aber nicht des des dir weiter helfen würde)

quakergod 12. Feb 2006 20:29

Re: Events und Threads in Schleife erzeugen
 
so, mein code sieht nun wie folgt aus:

Delphi-Quellcode:
type
  TgmDownload = record
    Event: TEvent;
    Thread: TevtThread;
  end;

var
  FgmDownload: array of TgmDownload;


procedure TfrmMain.Button5Click(Sender: TObject);
var dw: DWORD;
  EventList: array[0..1] of THandle; [b]// MUSS konstant sein da sonst WAIT_OBJECT_0 + Length(EventList) meckert! :([/b]
  i: integer;
begin
  FDLAbort := False;
  button1.Enabled := False;

  FDLThreads := strtoint(edtDLThreads.Text);
  SetLength(FgmDownload, FDLThreads); //Hier wird dann die Länge vom dyn. Array gesetzt

  //  SetLength(EventList, FDLThreads); // würde ich gerne machen :)

  for i := 0 to FDLThreads - 1 do
  begin
    FgmDownload[i].Event := TEvent.Create(nil, False, False, ''); //wird hier nämlich sowieso überschrieben
    FgmDownload[i].Thread := TEvtThread.Create(FgmDownload[i].Event);
    EventList[i] := FgmDownload[i].Event.Handle;
  end;
  try
    while FDLThreads > 0 do begin
         // warten auf Thread oder Input
      dw := MsgWaitForMultipleObjects(2, EventList, False,
        WF_TIMEOUT {INFINITE}, QS_ALLINPUT);
      case dw of
        WAIT_OBJECT_0 + 0: showmessage('1. Thread');
        WAIT_OBJECT_0 + 1: showmessage('2. Thread');
        WAIT_OBJECT_0 + Length(EventList): HandleDLUI;
        WAIT_TIMEOUT: HandleDlUI;
      else
         // Fehler auslösen (Notausgang)
        Break;
      end;
    end;
  finally
//    ev1.Free;
//    ev2.Free;
//    t1 := nil;
//    t2 := nil;
    showmessage('blubb');
    gridDL.DeleteRow(0);
    button1.Enabled := True;
  end;
end;
Nun habe ich folgendes Problem, wenn ich im edit "edtDLThreads" 1 Thread festlege passiert nichts, wenn ich eine 2 festlege macht er 2 Threads (beide Showmessages kommen) !?

Ausserdem kann ich das lokale array EventList nicht dynamisch machen weil er bei WAIT_OBJECT_0 + Length(EventList): HandleDLUI; einen Konstanten ausdruck braucht!

Vielen dank für deine Hilfe! :)

gsh 12. Feb 2006 20:49

Re: Events und Threads in Schleife erzeugen
 
des muss gehen hab ich schon gemacht mit Length der fehler muss wo anders liegen

Delphi-Quellcode:
dw := MsgWaitForMultipleObjects(2, EventList {kenn mich da nicht genau aus aber was muss man da angeben? den gibst da den Pointer vom Array an}, False, WF_TIMEOUT {INFINITE}, QS_ALLINPUT);

Kein Problem ich helfe gern

x000x 12. Feb 2006 22:17

Re: Events und Threads in Schleife erzeugen
 
Liste der Anhänge anzeigen (Anzahl: 1)
Moin moin,

so funktionierts, hab nur das von gsh eingebaut...
evtl. hilts ja weiter...


Alle Zeitangaben in WEZ +1. Es ist jetzt 11:06 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