![]() |
DLL und Callback-Funktion: Hostprogramm reagiert nicht?
Liste der Anhänge anzeigen (Anzahl: 1)
Hallo!
Ich probiere es gerade mit DLLs, indem ich tesweise ein Add-On für mein Programm erstelle. In der DLL habe ich einen Thread mit dem TCP-Client, der sich mit dem Remote-Server verbindet und sehr viele Daten empfängt. Jeder Datensatz wird einer Callback-Funktion (im Hostprogramm) als Parameter übergeben. Diese Funktion wandelt dann die Daten in einem Objekt um und zeigt die Daten an. Wie gesagt, das Ganze nur zum Zwecke des Lernens. Alles funktioniert ganz gut, solange die empfangenen Daten jeweils aus einem Datensatz bestehen und somit einzeln an die Callback-Funktion übergeben werden. Manchmal wenn der Datenaufkommen zu groß ist (oder warum auch immer) sendet der Server mehrere Datensätze auf einmal zu. In diesem Fall (um die Datenstau zu vermeiden) leitet der Client einen neuen Thread ab, der die Daten auffängt, aufteilt und einzeln der Callback-Funktion übergibt. Funktioniert auch. Das Problem tritt auf, wenn der Server mehrere "große" Datensätze hintereinander schickt und somit mehrere Threads hintereinander abgeleitet werden und sich gleichzeitig der Callback-Funktion bedienen. Dann friert das Hostprogramm plötzlich ein und reagiert nicht mehr. Es erscheint auch entsprechende Windows-Meldung, dass Programm nicht mehr reagiert. Gibt es etwas, was ich beachten soll wenn ich - Threads in der DLL erstelle - die Callback-Funktion aus mehreren Threads in der DLL gleichzeitig aufrufe? Sicherlich habe ich etwas nicht beachtet oder irgenwelche allgemeine Regel missachtet, die für soche Fälle üblich ist. Ich freue mich über jeden Hinweis. Vielen Dank! |
AW: DLL und Callback-Funktion: Hostprogramm reagiert nicht?
Zitat:
![]() |
AW: DLL und Callback-Funktion: Hostprogramm reagiert nicht?
Zitat:
Aber auch wenn ich eine ganz einfache Callback-Funktion erstelle, in der ich die Daten einfach in einem TMemo anzeigen lasse, tritt der Fehler trotzdem auf. |
AW: DLL und Callback-Funktion: Hostprogramm reagiert nicht?
Das Problem ist immer noch da :|
|
AW: DLL und Callback-Funktion: Hostprogramm reagiert nicht?
Zitat:
Hast du Fehler in deinen Threads denn auch schön ordentlich mit einem try..except außen herum abgefangen? Denn Exceptions in Threads, die nicht abgefangen werden, können sich genau so äußern. (Anderswo auch, aber in Threads ist das besonders schlecht.) Und dann gibt es da auch noch Tools wie MadExcept oder Eurekalog, die solche Exceptions abfangen und dir mitteilen wo das Problem aufgetreten ist. |
AW: DLL und Callback-Funktion: Hostprogramm reagiert nicht?
Zitat:
|
AW: DLL und Callback-Funktion: Hostprogramm reagiert nicht?
Ah. Du versuchst das Synchronize in der DLL. Das geht so nicht, denn da fehlt der Aufruf von CheckSynchronize. Das musst du z.B. über eine exportierte Funktion der DLL aus deinem Hauptprogramm heraus in ApplicationEvents.OnIdle aufrufen.
Da das bisher nicht aufgerufen wird, wird dein Synchronize nie abgearbeitet... |
AW: DLL und Callback-Funktion: Hostprogramm reagiert nicht?
Zitat:
Delphi-Quellcode:
, exportiere diese und rufe in der Hostaufendung einmal auf. Dann kann ich aus der DLL heraus mit dem hostprogramm synchronisiert. Richtig?
CheckSynchronize
|
AW: DLL und Callback-Funktion: Hostprogramm reagiert nicht?
Habe ich jetzt gemacht wie oben beschrieben. Hat sich nichts geändert. Die mit Synchronize augerufene Prozedur wird nicht ausgeführt.
|
AW: DLL und Callback-Funktion: Hostprogramm reagiert nicht?
Nochmal von vorne...
Alles richtig bis hier? Wenn ich zumindest das Problem verstehe, dann ist es die Tatsache, dass (der Thread aus) der DLL mit der ganzen VCL-Geschichte des Hauptprogramms nichts am Hut hat. Und somit beispielsweise auch erst garnichts im Kontext des Hauptthreads ausführen kann. Richtig? Ich würde die Datenablieferungsfunktion so bauen, dass hier Daten einfach nur abgelegt werden. Beispielsweise in einer verketteten Liste (natürlich entsprechend geschützt, z.B. mit einem kritischen Abschnitt). In deinem Hauptprogramm würde ich z.B. jede halbe Sekunde hingehen und schauen, ob mittlerweile neue Daten vorliegen. Wenn ja, kann man die ja anzeigen. |
AW: DLL und Callback-Funktion: Hostprogramm reagiert nicht?
Zitat:
Zitat:
Zitat:
Trotzdem würde ich gerne wissen, warum sich das Testprogramm aufhängt. |
AW: DLL und Callback-Funktion: Hostprogramm reagiert nicht?
Nur schnell zusammengestöpselt, ein Rumpf , Fehlerkorrektur erbeten ...
DLL
Delphi-Quellcode:
library Project2;
uses SysUtils, Classes; Type TCallBackProc=Procedure(info:PChar); TDummyThread=Class(TThread) Procedure Execute;override; End; {$R *.res} Var SaveExit:Pointer; TheCallBackProc:TCallBackProc; procedure LibExit;far; begin ExitProc := SaveExit; end; Procedure InitDLL(CallBackProc:TCallBackProc); var i:Integer; begin TheCallBackProc := CallBackProc; for I := 0 to 100 do with TDummyThread.Create(false) do FreeOnTerminate := true; end; exports InitDLL index 1; { TDummyThread } procedure TDummyThread.Execute; begin inherited; Sleep(Random(2000)); if Assigned(TheCallBackProc) then TheCallBackProc(PChar(Format('Irnedein Testtext %d',[ThreadID]))); end; begin SaveExit := ExitProc; ExitProc := @LibExit; end. Testprogramm
Delphi-Quellcode:
unit Unit3;
interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, Dialogs, StdCtrls; type TCallBackProc=Procedure(info:PChar); TInitDLL=Procedure (p:TCallBackProc); TForm3 = class(TForm) Memo1: TMemo; Button1: TButton; procedure Button1Click(Sender: TObject); procedure FormCreate(Sender: TObject); procedure FormDestroy(Sender: TObject); private { Private-Deklarationen } FH:Thandle; FInitDLL:TInitDLL; public { Public-Deklarationen } end; var Form3: TForm3; implementation {$R *.dfm} procedure SyncAddToMemo(const s:String); begin Form3.Memo1.Lines.add(s); Application.ProcessMessages; end; procedure CallBack(P:PChar); begin TThread.Synchronize(nil,Procedure begin SyncAddToMemo(P) end) end; procedure TForm3.Button1Click(Sender: TObject); begin if Assigned(FInitDLL) then FInitDLL(@CallBack); end; procedure TForm3.FormCreate(Sender: TObject); begin FH := LoadLibrary(Pchar(ExtractFilePath(Paramstr(0))+'Project2.dll')); if FH<>0 then FInitDLL := GetProcAddress(FH,'InitDLL'); end; procedure TForm3.FormDestroy(Sender: TObject); begin if FH<>0 then FreeLibrary(FH); end; end. |
AW: DLL und Callback-Funktion: Hostprogramm reagiert nicht?
@bummi
Mach aus dem
Delphi-Quellcode:
ein
TThread.Synchronize
Delphi-Quellcode:
und das
TThread.Queue
Delphi-Quellcode:
kannst du unter den Tisch fallen lassen ;)
Application.ProcessMessages
Obwohl ich sowieso nicht sehe, wozu das wirklich gut sein soll ... :gruebel: |
AW: DLL und Callback-Funktion: Hostprogramm reagiert nicht?
Hast du denn mal die Trial von Eurekalog oder MadExcept ausprobiert (installiert und für beide Projekte, also Hauptprogramm und DLL, aktiviert)? Du musst die ja nicht sofort kaufen, sondern kannst es erst einmal ausprobieren. Du wirst aber denke ich merken, dass die durchaus sehr hilfreich sind und ihr Geld wert sind.
Denn das Programm hängt sich deinem Screenshot zufolge ja nicht auf, sondern hat einfach eine unbehandelte Exception. Die findest du auch in den Dr. Watson Debuglogs, sofern die auf deinem PC aktiviert gelassen sind. Den Stacktrace bekommst du da aber nur in aufbereiteter Form, wenn du .dbg Dateien aus den Delphi-Debugdateien erzeugst. |
AW: DLL und Callback-Funktion: Hostprogramm reagiert nicht?
@SirRufo für Queue müsste ich den Thread als Object aus der DLL zurückgeben.
Der Hintergrund was eigentlich nur zu zeigen wo IMHO synchronisiert werden muss. |
AW: DLL und Callback-Funktion: Hostprogramm reagiert nicht?
Zitat:
Delphi-Quellcode:
vor)
class procedure
![]() ![]() |
AW: DLL und Callback-Funktion: Hostprogramm reagiert nicht?
@SirRufo ich lasse mich gerne belehren ich hatte den Fehler bei der Übergabe von nil auf den fehlenden Thread bezogen.
Du hast recht dieser muss nicht übergeben werden, trotzdem fliegt mir das ganze bei ausreichender Threadanzahl mit Queue um die Ohren, ich vermute weil der übergebene PChar bis zum abarbeiten der Queue ungültig geworden ist. |
AW: DLL und Callback-Funktion: Hostprogramm reagiert nicht?
Deshalb nimmt man da ja auch keinen PChar, sondern z.B. einen WideString, wenn etwas asynchron weiterverarbeitet wird.
|
AW: DLL und Callback-Funktion: Hostprogramm reagiert nicht?
Wenn der PChar in einer lokalen string Variable gespeichert wird dann sollte dieser auch problemlos per Queue weitergegeben werden können.
Widestring ist da natürlich wesentlich nkomplizierter :) |
AW: DLL und Callback-Funktion: Hostprogramm reagiert nicht?
Zitat:
Delphi-Quellcode:
Wo da wirklich der Speicher freigegeben wird, wäre ich mir jetzt nicht so sicher...
procedure TTest.Execute;
var a: PChar; begin a := PChar(DupeString('f', 30)); // um keine Konstante zu haben TThread.Queue(nil, procedure begin ShowMessage(a); // fdsdfsd end); a := 'fdsdfsd'; end; (und ich habe grad keine Lust zu schauen ;-)) |
Alle Zeitangaben in WEZ +1. Es ist jetzt 10:21 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