Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Programmieren allgemein (https://www.delphipraxis.net/40-programmieren-allgemein/)
-   -   Delphi ExecAndWait blockt alles...incl. Threads? (https://www.delphipraxis.net/215803-execandwait-blockt-alles-incl-threads.html)

haentschman 10. Sep 2024 05:42


ExecAndWait blockt alles...incl. Threads?
 
Liste der Anhänge anzeigen (Anzahl: 1)
Hallöle...8-)

Ich möchte währen der Validierung der X-Rechnung (kann bis zu 10 Sekunden dauern) dem User über eine Marquee Progressbar eine Rückmeldung geben. "Ich arbeite noch"

Gegeben:
In der X-Rechnung Unit gibt es für die Ausführung ein...
Zitat:

cmd.SaveToFile(tmpFilename+'.bat',TEncoding.ANSI);

Result := ExecAndWait(tmpFilename+'.bat',''); // int.XRechnungValidationHelperJava (214)

_CmdOutput := CmdOutput.Text;
Das die GUI nicht reagiert war logisch. Deshalb habe ich dem Progress Fenster einen Thread verpaßt, der ein ApplicationProcessMessages (Anweisung von DevExpress :?) während der Zeit regelmäßig ausführt.(Bild)

Problem:
Der Thread läuft...aber nur ein mal kommt der ins "Synchronize(SyncExecute)"...solange ExecAndWait noch nicht ausgeführt wird. Das Ganze ist dann komplett blockiert! Nach dem ExecAndWait macht der Thread weiter...aber zu spät. :?

Frage:
1. Kann ich die function TXRechnungValidationHelperJava.Validate( in einem Task/Thread zusätzlich ausführen?
2. Warum wird das ganze blockiert?

Danke...:wink:

jaenicke 10. Sep 2024 07:58

AW: ExecAndWait blockt alles...incl. Threads?
 
Der Thread kann sich mit dem Hauptthread nicht synchronisieren, wenn dieser durch die länger dauernde Operation blockiert ist. Du solltest stattdessen die blockierende Operation in einen Thread verlagern.

himitsu 10. Sep 2024 09:31

AW: ExecAndWait blockt alles...incl. Threads?
 
Jupp, es gibt nur zwei drei Wege, um sowas zu lösen:

* Die Arbeit im Hauptthread, dann kann die VCL dort halt so lange nichts machen. (nur Grafik/Wartemauszeiger zu Beginn, währenddessen halt unbewegt und zum Schluß wieder weg ... in der Hoffnung Windows reagiert nach standardmäßig 3 Sekunden nicht mit "Fenster reagiert nicht mehr")

* Die Arbeit in einem Thread, dann kann der Hauptthread in Ruhe die Wartegrafik aktualisieren
* Arbeit im Hauptthread, aber dort wird z.B. über einen Callback oder regelmäßig ProcessMessages der Wartegrafik Zeit gegeben.
* Oder Arbeit im Hauptthread und die Wartegrafik im Thread.



Wir blenden z.B. bei längeren Aktionen, bzw. nach kurzem Warten, ein Fenster (nicht VCL) ein, in einem anderen Thread erstellt und behandelt, wo per Button oder per Strg+ESC (GetAsyncKeyState) abgebrochen werden kann.
Unsere DB-Komponenten haben eine Methode, womit man sie aus einem anderen Thread heraus abbrechen kann (also wo versucht wird dem DB-Server ein Abbruchsignal zu senden, bzw. die Connection getrennt wird, wodurch die wartende Aktion im Hauptthread abbricht)

Ursprünglich auch versucht die DB-Aktion in den Thread zu verschieben und das WarteZeugs im Hauptthread, aber trotz viel Arbeit mit Blocking, ConnectionPooling uvm. hatte es immer wieder mal geknallt, da ja öfters VCL-Komponenten und Events ungeschützt an den Queries hingen.

haentschman 10. Sep 2024 11:10

AW: ExecAndWait blockt alles...incl. Threads?
 
Moin...8-)
Durch meine Erkältung bin grade gedanklich nicht so fit :vernupft:...Beim Drüberlesen, und dem Gespräch mit Tbx, war es eigentlich klar. ExecAndWait läuft ja im Hauphread. :duck:
Zitat:

Du solltest stattdessen die blockierende Operation in einen Thread verlagern.
...wird auch gemacht. :zwinker:

rebas 20. Sep 2024 03:29

AW: ExecAndWait blockt alles...incl. Threads?
 
Zitat:

Hallöle...

Ich möchte währen der Validierung der X-Rechnung (kann bis zu 10 Sekunden dauern) dem User über eine Marquee Progressbar eine Rückmeldung geben. "Ich arbeite noch"
snow rider 3d
In der X-Rechnung Unit gibt es für die Ausführung ein...
Zitat:
cmd.SaveToFile(tmpFilename+'.bat',TEncoding.ANSI);

Result := ExecAndWait(tmpFilename+'.bat',''); // int.XRechnungValidationHelperJava (214)

_CmdOutput := CmdOutput.Text;
Das die GUI nicht reagiert war logisch. Deshalb habe ich dem Progress Fenster einen Thread verpaßt, der ein ApplicationProcessMessages (Anweisung von DevExpress ) während der Zeit regelmäßig ausführt.(Bild)

Problem:
Der Thread läuft...aber nur ein mal kommt der ins "Synchronize(SyncExecute)"...solange ExecAndWait noch nicht ausgeführt wird. Das Ganze ist dann komplett blockiert! Nach dem ExecAndWait macht der Thread weiter...aber zu spät.

Frage:
1. Kann ich die function TXRechnungValidationHelperJava.Validate( in einem Task/Thread zusätzlich ausführen?
2. Warum wird das ganze blockiert?

Danke...
Task oder Thread: Du kannst die TXRechnungValidationHelperJava.Validate in einem separaten Task oder Thread ausführen, um die GUI nicht zu blockieren.

Async/Wait: Verwende einen asynchronen Ansatz mit Delphi's TTask oder TThread, um das Blocking zu vermeiden und die GUI responsive zu halten.
Code:
TTask.Run(procedure
begin
  Result := ExecAndWait(tmpFilename + '.bat', '');
  TThread.Synchronize(nil, procedure
  begin
    ProgressFenster.Close;
  end);
end);


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