AGB  ·  Datenschutz  ·  Impressum  







Anmelden
Nützliche Links
Registrieren

BeginThread - Methoden aufruf

Ein Thema von youuu · begonnen am 28. Aug 2010 · letzter Beitrag vom 30. Aug 2010
Antwort Antwort
youuu

Registriert seit: 2. Sep 2008
Ort: Kleve
822 Beiträge
 
Delphi 2010 Professional
 
#1

AW: BeginThread

  Alt 28. Aug 2010, 21:54
Ok nun verstehe ich es.

Was ist denn die beste Möglichkeit eine Procedure einer Klasse als Thread auf zu rufen?

1. Klasse ist schon created
2. es muss nur eine bestimmte Procedure der klasse abgearbeitet werden im Thread.
Steven
  Mit Zitat antworten Zitat
Benutzerbild von lbccaleb
lbccaleb

Registriert seit: 25. Mai 2006
Ort: Rostock / Bremen
2.037 Beiträge
 
Delphi 7 Enterprise
 
#2

AW: BeginThread - Methoden aufruf

  Alt 28. Aug 2010, 21:59
Wenn mich nicht alles täuscht muss man doch auf die function zeigen?? also

Delphi-Quellcode:
function Blub(Parameter: Pointer): integer;
begin
  // ...
end;

// Aufruf
BeginThread(nil, 0, [COLOR="Red"]@[/COLOR]Blub, nil, 0, ThreadID);
oder nicht??
Martin
MFG Caleb
TheSmallOne (MediaPlayer)
Die Dinge werden berechenbar, wenn man die Natur einer Sache durchschaut hat (Blade)
  Mit Zitat antworten Zitat
youuu

Registriert seit: 2. Sep 2008
Ort: Kleve
822 Beiträge
 
Delphi 2010 Professional
 
#3

AW: BeginThread - Methoden aufruf

  Alt 28. Aug 2010, 22:03
Was michauch irritiert, hier wird eine procedure verwendet.

hier
Steven
  Mit Zitat antworten Zitat
Benutzerbild von Björn Ole
Björn Ole

Registriert seit: 10. Jul 2008
166 Beiträge
 
Delphi XE Professional
 
#4

AW: BeginThread - Methoden aufruf

  Alt 28. Aug 2010, 22:07
lbccaleb, geht beides.

So mit einer Methode:
Delphi-Quellcode:
TClass = class
public
  class procedure Blub(Param: Pointer); static;
end;

// Aufruf:
with TClass.Create do
  BeginThread(nil, 0, @Blub, nil, 0, ThreadID);
Ob das so aber vernünftig ist, weiß ich nicht.

Add: SirThornberry und himitsu erklären das im oben genannten Thread wunderprächtig. Ob nun eine function oder procedure übergeben wird, ist im Endeffekt egal.

Geändert von Björn Ole (28. Aug 2010 um 22:14 Uhr)
  Mit Zitat antworten Zitat
youuu

Registriert seit: 2. Sep 2008
Ort: Kleve
822 Beiträge
 
Delphi 2010 Professional
 
#5

AW: BeginThread - Methoden aufruf

  Alt 28. Aug 2010, 22:13
Hm, dann habe ich dieses Problem

Instanzenelement 'FslKat' in diesem Zusammenhang nicht verfügbar //FslKat ist eine StringList aus der Klasse im privat Bereich
Steven

Geändert von youuu (28. Aug 2010 um 22:16 Uhr)
  Mit Zitat antworten Zitat
Benutzerbild von Björn Ole
Björn Ole

Registriert seit: 10. Jul 2008
166 Beiträge
 
Delphi XE Professional
 
#6

AW: BeginThread - Methoden aufruf

  Alt 28. Aug 2010, 22:29
Siehe http://www.delphipraxis.net/1032983-post3.html
Zitat von himitsu:
> statische (static) Klassen-Methode: Self gibt's nicht
Darum kannst du auch nicht auf die Felder deiner Klasse zugreifen.


Weitere Möglichkeit, allerdings darf hier die Methode, die du BeginThread übergibst, keine Parameter entgegennehmen, da dieser vom unsichtbaren "Self" belegt wird.

Delphi-Quellcode:
TClass = class
public
  class procedure Blub;
end;

var
  ThreadID: Cardinal;
  p: procedure of object;
begin
  with TClass.Create do
    p := Blub;
  BeginThread(nil, 0, p, nil, 0, ThreadID);
  Mit Zitat antworten Zitat
youuu

Registriert seit: 2. Sep 2008
Ort: Kleve
822 Beiträge
 
Delphi 2010 Professional
 
#7

AW: BeginThread - Methoden aufruf

  Alt 28. Aug 2010, 22:48
Was gäbe es für eine Möglichkeit ohne BeginThread?
Denn diese scheint mir sehr suboptimal, für mein Problem.

Ich muss diese Methode MyClass.Scan(o); in einen Thread abarbeiten lassen, nur wie?
Der Parameter muss unbedingt übergeben werden, sowie aus der Klasse entsprechende Variablen benutzbar bleiben.
Steven

Geändert von youuu (28. Aug 2010 um 22:55 Uhr)
  Mit Zitat antworten Zitat
Benutzerbild von Luckie
Luckie

Registriert seit: 29. Mai 2002
37.621 Beiträge
 
Delphi 2006 Professional
 
#8

AW: BeginThread - Methoden aufruf

  Alt 29. Aug 2010, 12:24
...weil du mit eine Klassen Prozedur einer Klasse keinen Zugriff auf die anderen Elemente einer Klasse hast. Klassen Prozeduren funktionieren ohne eine Instanz der Klasse.
Michael
Ein Teil meines Codes würde euch verunsichern.
  Mit Zitat antworten Zitat
Benutzerbild von littleDave
littleDave

Registriert seit: 27. Apr 2006
Ort: München
556 Beiträge
 
Delphi 7 Professional
 
#9

AW: BeginThread - Methoden aufruf

  Alt 29. Aug 2010, 20:16
Mit ein paar Tricks kann man auch Methoden mit Parametern, ohne "große" Probleme bekommen.

Delphi-Quellcode:
interface

type
  // Event-Deklaration, die in der Klasse aufgerufen wird
  TThreadExecuteEvent = procedure(const Params: array of const) of object;

procedure RunInThread(Meth: TThreadExecuteEvent; const Params: array of const); register;

implementation

type
  // interne Daten
  TConstArray = array of TVarData;
  PConstArray = ^TConstArray;

  PExecutionInfo = ^TExecutionInfo;
  TExecutionInfo = packed record
    EntryPoint : TThreadExecuteEvent;
    ArrayLen : integer;
    Params : PConstArray;
  end;

// Dass ist die Methode, die man bei "BeginThread" aufruft, als Parameter bei "BeginThread"
// kommt dann der Pointer auf ein TExecutionInfo - record
function __internThreadProc(ExecInfo: PExecutionInfo): integer; stdcall;
var pPtr : Pointer;
    pSelf : Pointer;
    pBase : Pointer;
    len : integer;
begin
  try
    // Start Address of the method
    pPtr := TMethod(ExecInfo^.EntryPoint).Code;
    // Self-Pointer
    pSelf := TMethod(ExecInfo^.EntryPoint).Data;
    // Parameter array
    pBase := ExecInfo^.Params;
    // length of Parameter
    len := ExecInfo^.ArrayLen;
    asm
      // save base registers
      push ecx
      push ebx
      push eax

      // push self ptr
      mov eax, pSelf
      // push array of const
      // array of const ->
      // 1st the pointer to the 1st element of the array
      mov ebx, pBase
      // 2nd (hidden parameter): high value of the array
      mov ecx, len

      // call
      call pPtr

      // restore registers
      pop eax
      pop ebx
      pop ecx
    end;
  finally
    // ExecInfo freigeben (wurde von "CallMethod" erzeugt)
    Dispose(ExecInfo);
  end;
  result := 0;
end;

procedure RunInThread(Meth: TThreadExecuteEvent; const Params: array of const);
var p: PExecutionInfo;
begin
  // p wird in der Methode "__internThreadProc" wieder freigegeben
  New(p);
  p^.EntryPoint := Meth;
  p^.ArrayLen := High(Params);
  p^.Params := PConstArray(@Params);

  // hier der BeginThread - Aufruf:
  // BeginThread(nil, 0, @__internThreadProc, p, 0, ThreadID);
  
  // Test-Code: direkter Aufruf
  __internThreadProc(p);
end;
Hier mal ein kleines Beispiel zur Benutzung:
Delphi-Quellcode:
type
  TTest = class
  private
    FValue : string;
  public
    procedure Test(const Params: array of const);
  end;

{ TTest }

procedure TTest.Test(const Params: array of const);
begin
  // Test des Self-Pointers
  ShowMessage(FValue);
  // Test auf Länge des Params-Array
  ShowMessage(IntToStr(length(Params)));
  // Ausgabe des 1. Parameters
  ShowMessage(IntToStr(TVarRec(Params[0]).VInteger));
end;

procedure TForm1.Button1Click(Sender: TObject);
var t: TTest;
begin
  t := TTest.Create;
  try
    t.FValue := 'Hallo';
    RunInThread(t.Test, [12345, 15676]);
  finally
    // wichtig: für das Testen wird kein Thread erstellt
    // daher können wir im finally-Part auch das Objekt wieder freigeben.
    // Falls man nun wirklich die Sache in einem Thread ausführt, darf
    // das Objekt erst freigegeben werden, wenn der Thread beendet ist - also nicht
    // hier.
    t.Free;
  end;
end;
Das ganze ist nur als Denk-Anstoß gedacht.

Gruß

EDIT
Ich seh gerade, dass das ganze per Threading noch nicht so ganz läuft: der Hacken ist in der Methode "RunInThread" in der Zeile: p^.Params := PConstArray(@Params); . Beim direkten Aufruf (also der Testcode), fällt der Fehler noch nicht auf. Jedoch wenn man es per Threading macht: das array-of-const wird freigegeben, sobald die Methode verlassen wird. Jedoch ist der Thread mit 99%er wahrscheinlich noch nicht beendet, d.h. der Pointer auf das "array-of-const"-Array wird ungültig. Man müsste das Array in der Methode neu kopieren - aber darauf habe ich gerade keine Lust mehr . Außerdem ist es wirklich etwas übertrieben. Hier mal eine einfache "RunInThread"-Methode, die ein beliebiges Objekt als Parameter entgegen nimmt (damit sollte es relativ einfach sein, das Grundprinzip zu verstehen):

Delphi-Quellcode:
interface

procedure RunInThread(Meth: TNotifyEvent; const Params: TObject);

implementation

type
  // interne Daten
  PExecutionInfo = ^TExecutionInfo;
  TExecutionInfo = packed record
    EntryPoint : TNotifyEvent;
    Data : TObject;
  end;

// Dass ist die Methode, die man bei "BeginThread" aufruft, als Parameter bei "BeginThread"
// kommt dann der Pointer auf ein TExecutionInfo - record
function __internThreadProc(ExecInfo: PExecutionInfo): integer; stdcall;
begin
  try
    ExecInfo^.EntryPoint(ExecInfo^.Data);
  finally
    // ExecInfo freigeben (wurde von "CallMethod" erzeugt)
    Dispose(ExecInfo);
  end;
  result := 0;
end;

procedure RunInThread(Meth: TThreadExecuteEvent; const Data: TObject);
var p: PExecutionInfo;
begin
  // p wird in der Methode "__internThreadProc" wieder freigegeben
  New(p);
  p^.EntryPoint := Meth;
  p^.Data := Data;

  // hier der BeginThread - Aufruf:
  // BeginThread(nil, 0, @__internThreadProc, p, 0, ThreadID);
  
  // Test-Code: direkter Aufruf
  __internThreadProc(p);
end;
Noch ein Beispiel
Delphi-Quellcode:
type
  TTest = class
  private
    FValue : string;
  public
    procedure Test(Sender: TObject);
  end;

{ TTest }

procedure TTest.Test(Sender: TObject);
begin
  // Test des Self-Pointers
  ShowMessage(FValue);
  // Test des Parameters
  ShowMessage(TTest(Sender).FValue);
end;

procedure TForm1.Button1Click(Sender: TObject);
var t: TTest;
begin
  t := TTest.Create;
  try
    t.FValue := 'Hallo';
    RunInThread(t.Test, t);
  finally
    // wichtig: für das Testen wird kein Thread erstellt
    // daher können wir im finally-Part auch das Objekt wieder freigeben.
    // Falls man nun wirklich die Sache in einem Thread ausführt, darf
    // das Objekt erst freigegeben werden, wenn der Thread beendet ist - also nicht
    // hier.
    t.Free;
  end;
end;
Jabber: littleDave@jabber.org
in case of 1 is 0 do external raise while in public class of object array else repeat until 1 is 0

Geändert von littleDave (29. Aug 2010 um 21:07 Uhr)
  Mit Zitat antworten Zitat
Benutzerbild von sx2008
sx2008

Registriert seit: 15. Feb 2008
Ort: Baden-Württemberg
2.332 Beiträge
 
Delphi 2007 Professional
 
#10

AW: BeginThread - Methoden aufruf

  Alt 30. Aug 2010, 00:01
@littleDave: Interessanter Code; aber ich würde das nicht verwenden.
Es gibt wohl niemand hier der den Sourcecode (incl. Assembler) wirklich bis in die Tiefe versteht.
Man sollte nicht möglichst clever einen Thread starten, sondern der Code sollte möglichst einfach und leicht verständlich sein. ==> TThread-Klasse verwenden!
Hier noch ein Artikel (englisch), der zeigt, dass man nicht clever, sondern einfach programmieren sollte.
  Mit Zitat antworten Zitat
Antwort Antwort

Themen-Optionen Thema durchsuchen
Thema durchsuchen:

Erweiterte Suche
Ansicht

Forumregeln

Es ist dir nicht erlaubt, neue Themen zu verfassen.
Es ist dir nicht erlaubt, auf Beiträge zu antworten.
Es ist dir nicht erlaubt, Anhänge hochzuladen.
Es ist dir nicht erlaubt, deine Beiträge zu bearbeiten.

BB-Code ist an.
Smileys sind an.
[IMG] Code ist an.
HTML-Code ist aus.
Trackbacks are an
Pingbacks are an
Refbacks are aus

Gehe zu:

Impressum · AGB · Datenschutz · Nach oben
Alle Zeitangaben in WEZ +1. Es ist jetzt 06:02 Uhr.
Powered by vBulletin® Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024-2025 by Thomas Breitkreuz