![]() |
Probleme mit PostMessage aus einem Thread
Hi,
sporadisch habe ich das Problem, dass mein Zeichnen der Controls auf meinem Formular Exceptions auftreten. Nach einiger Suche bin ich auf die vermutlichen Ursache gestoßen: Ich habe ein Control, abgeleitet vom VirtualStringTree. Dieses Control besitzt eine Methode, die aus einem anderen Thread aufgerufen wird, um dem Control mitzuteilen, dass es sich aktualisieren soll. Um nicht direkt aus dem anderen Thread zu zeichnen, gehe ich wie folgt vor:
Delphi-Quellcode:
Außerdem gibt es noch eine Methode, die auf diese Nachricht reagiert:
procedure MyControl.DoCallback;
begin PostMessage(Self.Handle, WM_MYCONTROLRELOAD, 0, 0); end;
Delphi-Quellcode:
Das Problem ist nun, dass die Zeichenroutine des Controls aus dem Thread aufgerufen wird, der auch die Callback aufgerufen hat. Bisher war ich davon ausgegangen, dass wenn ich eine Nachricht mit PostMessage an mein Control schicke, diese auch im Hauptthread abgearbeitet wird. Das ist anscheinend nicht der Fall?!
procedure DoControlReload( var Message: TMessage); message WM_MYCONTROLRELOAD;
procedure MyControl.DoControlReload( var Message: TMessage); begin ReloadfromDB; Invalidate; end; Zusätzlich möchte ich noch erwähnen, dass der Thread im Execute eine while-Schleife beinhaltet, in der auch Application.ProcessMessages aufgerufen wird. Meine Frage an Euch: Wie kann ich nun verhindern, dass die Zeichenroutine des Controls aus dem anderen Thread aufgerufen wird? Viele Grüße, deadcantdance |
Re: Probleme mit PostMessage aus einem Thread
Nimm das Application.ProcessMessages aus dem Thread raus :stupid:
|
Re: Probleme mit PostMessage aus einem Thread
Das kann ich nicht, der Thread läuft mit höherer Priorität, da es sich um den Kommunikationsthread handelt. Nehme ich die Anweisung raus, friert mir die GUI ein.
|
Re: Probleme mit PostMessage aus einem Thread
Bau ein sleep(0) ein!
|
Re: Probleme mit PostMessage aus einem Thread
Hilft leider auch nicht, sobald ich in der GUI auf einen Button klicke, friert diese ein!
|
Re: Probleme mit PostMessage aus einem Thread
Und "SwitchToThread;"?
Edit: Du solltest wahrscheinlich eher die Priorität etwas zurückschrauben. |
Re: Probleme mit PostMessage aus einem Thread
Ich muss mich korrigieren, die Priority steht auf Normal.
SwitchToThread hilft leider auch nicht, CPU-Auslastung geht auf 100% |
Re: Probleme mit PostMessage aus einem Thread
Kein Ahnung, was du da machst.
Self.Handle ist auch noch ein Problem (aber ich vermute nicht das primäre). Besser ist du übergibst das Handle an das Threadobjekt und machst dort Postmessage. Dann vielleicht mal sleep(x) (mit x>0). Damit hier mal etwas pausiert. |
Re: Probleme mit PostMessage aus einem Thread
Application.ProcessMessages darf nicht im Kontext des Threads laufen; da beisst keine Maus einen Faden ab!
Dein Thread sollte auch nicht unbedingt mit höherer Prio laufen. Wenn dein Thread auf die Benutzeroberfläche zugreifen muss, dann immer mit der Methode Synchronize:
Delphi-Quellcode:
Der Trick ist nun, dass Synchronize eine Windows Botschaft in die Messagequeue setzt
procedure TMyThread.Execute;
begin for i := ... to ... do begin FStatus := 'Lese '+ IntToStr(i); Synchronize(UpdateLabel); DatenLesen; FStatus := 'Verarbeite '+ IntToStr(i); Synchronize(UpdateLabel); Datenverarbeiten; Synchronize(UpdateGui); if Terminated then Exit; // wichtig, um den Thread sauber abbrechen zu können end; end; procedure TMyThread.UpdateLabel; begin form1.StatusLabel.Caption := FStatus; form1.StatusLabel.Refresh; end; procedure TMyThread.UpdateGui; begin form1.vsg1.items.Add(....); form1.vsg1.refresh; end; und die Methode UpdateGui im Kontext der Hauptthreads ausgeführt wird. Über Synchronize kannst du nur eine Methode ohne Parameter aufrufen. Diese Einschränkung lässt sich umgehen, indem die Parameter als Variablen im Thread-Objekt gespeichert werden. Übrigens: Wenn dein Thread niemals wartet (WaitforMultipleObjects), dann ist es ganz normal, dass die CPU-Auslastung auf 100% geht! |
Re: Probleme mit PostMessage aus einem Thread
Zitat:
Ich dachte immer, Application.Processmessages erzwingt die Abarbeitung der Windows-Eventqueue. Das kann doch auch aus einem Thread sinnvoll sein, z.B. wenn es Zugriffe auf Datenbanken, Hardware-dlls etc. gibt. Grüße, Messie |
Alle Zeitangaben in WEZ +1. Es ist jetzt 08:51 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