AGB  ·  Datenschutz  ·  Impressum  







Anmelden
Nützliche Links
Registrieren
Thema durchsuchen
Ansicht
Themen-Optionen

Thread, Synchronize, MessageDlg & Interaktion

Ein Thema von Timelesk · begonnen am 20. Jul 2010 · letzter Beitrag vom 20. Jul 2010
Antwort Antwort
Timelesk

Registriert seit: 24. Jul 2004
72 Beiträge
 
#1

Thread, Synchronize, MessageDlg & Interaktion

  Alt 20. Jul 2010, 17:13
Hi,

ich hantiere momentan mit Threads herum und bin dabei auf ein Problem gestoßen, welches ich nicht beheben kann.

Aufgabe
Ich lader per Thread Informationen von einer Website und möchte, dass bei einer Fehlermeldung (Ausgabe der Website) der Benutzer per MessageDlg interagieren kann.
Hierbei wählt er ob er sich erneut anmelden möchte.

Ablauf
Delphi-Quellcode:
procedure UpdateError();
var
  Response: Integer;
begin
  Response := MessageDlg('Erneut anmelden?', mtWarning, [mbYes,mbNo], 0);
  if Response = mrYes then
    begin
      Main.StartThread('login');
    end
  else
    begin
      Main.Pager.ActivePage := 3;
    end;
end;

function getSourcecode(): Boolean
begin
  //Lade Ausgabe von Website
  //...
  //Prüfe ob die Ausgabe erfolgreich war
  Result := checkSourcecode(Sourcecode);
  if not(Result)
    begin
      Self.Terminate;
      Synchronize(UpdateError);
    end;
end;
Fragen
1) Wenn ich das Terminate in die Procedure UpdateError reinpacke, worauf bezieht sich dann das Self? Immernoch auf den Thread oder auf den Thread des Hauptformulars? Weil durch Synchronize wird ja die Funktion im Hauptthread ausgeführt, oder?

2) Ich habe festgestellt, dass der SubThread wartet, bis die Procedure UpdateError vollständig ausgeführt wurde. D.h. bis mein MessageDlg bestätigt wurde. Kann man das umgehen? Also der Thread soll dann einfach "auslaufen", sich beenden. Er hat seine Arbeit getan, er soll halt nur noch den MessageDlg anzeigen und gut ist.

Momentan ist es so, dass sich das Programm vollständig aufhängt sobald ich auf Ja klicke.
Ich kann 20x auf Nein klicken und dann auf Ja und es friert vollständig ein.
Wenn ich direkt nach Response := MessageDlg(...); ein exit; einbaue, so funktioniert es jedoch.
D.h. es muss irgendetwas mit dem Beenden & Starten der Threads zu tun haben.

Die Funktion StartThread(String) prüft übrigens im Vorfeld, ob der Thread aktiv ist und beendet ihn.


Hoffe ich konnte mein Problem klar vorbringen und hoffe es gibt auch eine Lösung für mein Problem

Freue mich schon auf eure Antworten.
Vielen Dank.


lg
Timelesk
  Mit Zitat antworten Zitat
Benutzerbild von jfheins
jfheins

Registriert seit: 10. Jun 2004
Ort: Garching (TUM)
4.579 Beiträge
 
#2

AW: Thread, Synchronize, MessageDlg & Interaktion

  Alt 20. Jul 2010, 17:32

Fragen
1) Wenn ich das Terminate in die Procedure UpdateError reinpacke, worauf bezieht sich dann das Self? Immernoch auf den Thread oder auf den Thread des Hauptformulars? Weil durch Synchronize wird ja die Funktion im Hauptthread ausgeführt, oder?
Ja, wird im Hauptthread ausgeführt, aber Self bezieht sich immer auf die aktuelle Objektinstanz, also auf den Thread.

Zitat:
2) Ich habe festgestellt, dass der SubThread wartet, bis die Procedure UpdateError vollständig ausgeführt wurde. D.h. bis mein MessageDlg bestätigt wurde. Kann man das umgehen? Also der Thread soll dann einfach "auslaufen", sich beenden. Er hat seine Arbeit getan, er soll halt nur noch den MessageDlg anzeigen und gut ist.
Dann ist MessageDlg das falsche. Denn das wird afaik immer modal angezeigt. Du könntest auch eine Message an das Hauptformulat senden, und danach beenden. der Hauptthread zeigt dann die Dialogbox an.
Zitat:
Momentan ist es so, dass sich das Programm vollständig aufhängt sobald ich auf Ja klicke.
Ich kann 20x auf Nein klicken und dann auf Ja und es friert vollständig ein.
Verständlich. Denn wenn die Box angezeigt wird, ist der Thread noch nicht beendet. Und dann wird wieder StartThread aufgerufen (wohlgemerkt, der Thread ist wegen synchronize angehalten, ist aber noch aktiv.) und der Thread soll terminiert werden. Der antwortet aber nicht, weil er auf die Beendigung von StartThread wartet.
Klassischer Deadlock
  Mit Zitat antworten Zitat
Timelesk

Registriert seit: 24. Jul 2004
72 Beiträge
 
#3

AW: Thread, Synchronize, MessageDlg & Interaktion

  Alt 20. Jul 2010, 17:40
Hi,

vielen Dank für deine Antwort.

Wie meinst du das mit Message an das Hauptformular senden?
Ich seh da momentan nur die Möglichkeit nen Timer zu aktivieren, der halt nach 500ms ausgeführt wird und hoffe dass der Thread dann beendet ist. (ist aber definitiv keine Lösung)

gruß
Timelesk
  Mit Zitat antworten Zitat
Benutzerbild von jfheins
jfheins

Registriert seit: 10. Jun 2004
Ort: Garching (TUM)
4.579 Beiträge
 
#4

AW: Thread, Synchronize, MessageDlg & Interaktion

  Alt 20. Jul 2010, 17:51
Naja, so wie man halt Messages sendet

1. Du brauchst im Thread das Handle des Formulars. z.B. indem du ein public-Feld machst, und das Formular tut dann sein Handle da rein.
2. Nachricht definieren:const WM_ThreadFinished = WM_USER + 1234; 3. Wenn der Thread fertig ist:
Delphi-Quellcode:
  Result := checkSourcecode(Sourcecode);
  if not(Result)
    begin
      Self.Terminate;
      PostMessage(FormHandle, WM_ThreadFinished, Integer(Result), 0);
    end;
Im Hauptthread kannst du dann eine Methode implementieren, die genau auf diese Nachricht reagiert (message-Direktive) und da dann die Dialogbox darstellen und den Thread neu starten (vorher warten bis der alte abgearbeitet ist)


Alternativ kannst du auch einen neuen Thread kreieren ( halt MyThread = TMyThread.Create() ) und immer Freeonterminate auf true setzen. Dann startest du nicht einen Thread immer neu, sondern kreiert immer neue Threads, die sich nicht ins Gehege kommen. Dann kannst du dir den Message-Kram sparen - das ist definitiv die einfachere Lösung weil man sich um die Freigabe des Threads nicht mehr kümmern muss

Geändert von jfheins (20. Jul 2010 um 17:53 Uhr)
  Mit Zitat antworten Zitat
Timelesk

Registriert seit: 24. Jul 2004
72 Beiträge
 
#5

AW: Thread, Synchronize, MessageDlg & Interaktion

  Alt 20. Jul 2010, 18:56
Hi,

vielen Dank für deine Hilfe.
Nun funktioniert alles so, wie ich es möchte.

Die Message-Direktiven sind wirklich sehr nützlich.
Das Formular hatte ich bereits an den Thread weitergegeben und somit besaß ich auch das Handle.

Ganz verstanden hab ich das jetzt aber noch nicht.

Ich habe bei PostMessage 4 Parameter: Form-Handle, Message-Code, Param1, Param2
Wobei Param nachrichtenspezifische Werte sind.
Können die jeden Variablen-Typ annehmen? (Integer hast du ja oben geschrieben, was ist mit String, etc.?)

Des Weiteren heißen meine Procedures folgendermaßen: procedure WMLogin(var msg: TMessage); message WM_Login; Warum muss der erste Parameter TMessage sein? Sind die 2 nachrichtenspezifischen Parameter in TMessage enthalten?

gruß
Timelesk
  Mit Zitat antworten Zitat
Benutzerbild von jfheins
jfheins

Registriert seit: 10. Jun 2004
Ort: Garching (TUM)
4.579 Beiträge
 
#6

AW: Thread, Synchronize, MessageDlg & Interaktion

  Alt 20. Jul 2010, 19:03
Hi,

vielen Dank für deine Hilfe.
Nun funktioniert alles so, wie ich es möchte.
Schön zu hören
Zitat:
Ich habe bei PostMessage 4 Parameter: Form-Handle, Message-Code, Param1, Param2
Wobei Param nachrichtenspezifische Werte sind.
Können die jeden Variablen-Typ annehmen? (Integer hast du ja oben geschrieben, was ist mit String, etc.?)
Nein, die können jeweils nur 4 Byte aufnehmen (bzw. 8 Byte in x64 Anwendungen) - damit kann man wunderbar einen Integer übertragen, aber für Strings muss man dann Pointer hernehmen.
Zitat:
Des Weiteren heißen meine Procedures folgendermaßen: procedure WMLogin(var msg: TMessage); message WM_Login; Warum muss der erste Parameter TMessage sein? Sind die 2 nachrichtenspezifischen Parameter in TMessage enthalten?
1. Weil Delphi das so will, 2. Ja. TMesdsage ist ein Record in dem die Parameter drin sind. Oben habe ich Result nach Integer gecastet und in den 1. Parameter gesteckt. Den kannst du hier wieder zurück casten und auswerten (wenn du möchtest)
  Mit Zitat antworten Zitat
Timelesk

Registriert seit: 24. Jul 2004
72 Beiträge
 
#7

AW: Thread, Synchronize, MessageDlg & Interaktion

  Alt 20. Jul 2010, 19:05
Hi,

vielen Dank für deine Hilfe.
Hab ein wenig mit dem Record rumgespielt und funktioniert wunderbar.

Vielen Dank noch einmal

gruß
Timelesk
  Mit Zitat antworten Zitat
Benutzerbild von Luckie
Luckie

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

AW: Thread, Synchronize, MessageDlg & Interaktion

  Alt 20. Jul 2010, 21:27
Du könntest auch eine Message an das Hauptformulat senden,
Wenn man schon so schön mit Klassen und Objekten arbeitet, warum dann kein Ereignis auslösen?
Michael
Ein Teil meines Codes würde euch verunsichern.
  Mit Zitat antworten Zitat
mjustin

Registriert seit: 14. Apr 2008
3.006 Beiträge
 
Delphi 2009 Professional
 
#9

AW: Thread, Synchronize, MessageDlg & Interaktion

  Alt 20. Jul 2010, 21:35
Du könntest auch eine Message an das Hauptformulat senden,
Wenn man schon so schön mit Klassen und Objekten arbeitet, warum dann kein Ereignis auslösen?
Der Haken ist hier, dass die Aktion den laufenden Thread nicht blockieren darf. Messages sind daher ein Weg, weil sie in eine Warteschlange gestellt werden und der Thread dann sofort weiterläuft (und im konkreten Beispiel terminiert).
Ein Ereignis würde normalerweise einen Ereigniscode blockierend ausführen.
Ein Mittelweg wäre eine threadsichere Queue, in die man dann aus dem Ereigniscode eine Nachricht stellen kann. Das Ereignis wäre dann sofort abgearbeitet, der Thread läuft wie gewollt sofort weiter. Im Hauptthread kann die Queue dann z.B. durch einen Timer regelmäßig ausgelesen werden.
Michael Justin
habarisoft.com
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu
Online

Registriert seit: 11. Okt 2003
Ort: Elbflorenz
44.063 Beiträge
 
Delphi 12 Athens
 
#10

AW: Thread, Synchronize, MessageDlg & Interaktion

  Alt 20. Jul 2010, 22:20
Messages sind daher ein Weg, weil sie in eine Warteschlange gestellt werden und der Thread dann sofort weiterläuft.
Dieses gilt nur für PostMessage und PostThreadMessage, aber nicht in Verbindung mit SendMessage.
Neuste Erkenntnis:
Seit Pos einen dritten Parameter hat,
wird PoSex im Delphi viel seltener praktiziert.
  Mit Zitat antworten Zitat
Antwort Antwort


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 20:47 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