Delphi-PRAXiS
Seite 1 von 2  1 2      

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Win32/Win64 API (native code) (https://www.delphipraxis.net/17-win32-win64-api-native-code/)
-   -   Delphi Dynamisches Array in DLL kann nicht erweitert werden (https://www.delphipraxis.net/191814-dynamisches-array-dll-kann-nicht-erweitert-werden.html)

devidespe 22. Feb 2017 16:39

Dynamisches Array in DLL kann nicht erweitert werden
 
Hallo,

ich habe in meinem DLL-Code eine Prozedur, die ein dynamisches Array zurückliefern soll.

Die Prozedur beginnt mit:

Delphi-Quellcode:
procedure GetValues(Config : ConfigRec; var Ergebnis : Array of SingleDataSet);
und SingleDataSet ist deklariert als:

Delphi-Quellcode:
type SingleDataSet = packed record
       DataSetNr   : Byte;
       DataSetName : string[30];
       DataSetValue : Integer;
       SavingTime  : TDateTime;
     end;
Das kompiliert auf den ersten Blick fehlerfrei durch.

Wenn ich allerdings in der selben Prozedur das Array erweitern möchte, um Daten anzufügen, scheitert dies:

Delphi-Quellcode:
SetLength(Ergebnis, 2);
mit Fehler E2008 Inkompatible Typen. Wie kann ich das trotzdem bewerkstelligen in Verbindung mit einem dynamischen Array?

Fritzew 22. Feb 2017 16:56

AW: Dynamisches Array in DLL kann nicht erweitert werden
 
Standardmässig benutzen Programm und Dll verschiedene Speichermanager.
Das bedeutet das in einer DLL Schnittstelle nur einfache typen wie integer, double, Pchar, etc verwendet werden sollten.
Du kannst die Unit Sharemem mit einbinden. Funktioniert aber nur wenn die DLL und das Programm mit Delphi kompiliert werden.
Das nächste Problem:
die Deklaration von
Delphi-Quellcode:
var Ergebnis : Array of SingleDataSet
Das ist ein OpenArray. http://docwiki.embarcadero.com/RADSt...eters_(Delphi)

Am besten Du deklarierst einen Typ für das Array:

Delphi-Quellcode:
type tSingleDatasetArray = Array of SingleDataSet;
Ich wäre aber sehr vorsichtig mit solchen Parametern über eine Dll.
Am besten stelle auf packages um. (dafür sind diese gedacht).
Da die Dll ja sowieso Delphi Only ist mit solchen Parameter würde ich diesen Weg gehen.

Sailor 22. Feb 2017 16:59

AW: Dynamisches Array in DLL kann nicht erweitert werden
 
Array of SingleDataSet ist ein OpenArray-Parameter, kein dynamisches Feld.
Richtig geht es so:
Delphi-Quellcode:
TSingleDataSetArray = Array of SingleDataSet;

procedure GetValues(var Ergebnis:TSingleDataSetArray);

Aviator 23. Feb 2017 08:48

AW: Dynamisches Array in DLL kann nicht erweitert werden
 
Ich würde die Länge des Arrays auch schon vor dem Aufruf setzen. Da es sich hier ja um einen Referenzparameter handelt, könntest du dir noch eine zusätzliche Funktion bauen, die dir die benötigte Länge zurückgibt.

Sowas in der Art:

Delphi-Quellcode:
procedure SomeMethod;
var
  Count: Integer;
  MyArray: TSingleDataSetArray;
begin
  if MyDllInstance.GetElementCount(Count) then begin
    SetLength(MyArray, Count);
    GetValues(MyArray);
  end;
end;
So habe ich es in meinen DLLs aufgebaut und das funktioniert wunderbar. Erinnert in abgewandelter Form auch an die Win32API bei der das ähnlich gelöst wird. Erst die Methode mit dem Referenzparamenter nil aufrufen um sich die benötigte Länge zu holen, dann die Länge des Arrays setzen und dann die Methode mit dem richtigen Array Parameter wieder ausführen.

devidespe 23. Feb 2017 15:25

AW: Dynamisches Array in DLL kann nicht erweitert werden
 
Zitat:

Zitat von Sailor (Beitrag 1362351)
Array of SingleDataSet ist ein OpenArray-Parameter, kein dynamisches Feld.
Richtig geht es so:
Delphi-Quellcode:
TSingleDataSetArray = Array of SingleDataSet;

procedure GetValues(var Ergebnis:TSingleDataSetArray);

Hallo und erstmal danke für die Rückmeldung. Dein Ansatz funktioniert auf den ersten Blick, aber sobald die Prozedur beginnt, erhalte ich eine Exception bzw. "Access Violation read of address 0xffffffff". Daher habe ich einfacherweise mal den Record weggelassen, und folgendes deklariert:

Delphi-Quellcode:
type TIntegerArray = Array of integer;
...
procedure GetValues(Config : ConfigRec; var Ergebnis : TIntegerArray);
...
leider mit dem gleichen Ergebnis. Innerhalb der Prozedur mache ich noch nichtmal was mit dem "Ergebnis", weder Größenanpassungen noch sonst was. Aber es wird leider jedesmal mit der Exception quittert.

hoika 23. Feb 2017 15:41

AW: Dynamisches Array in DLL kann nicht erweitert werden
 
Hallo,
hast du denn ShareMem eingebunden?

Aviator 23. Feb 2017 16:28

AW: Dynamisches Array in DLL kann nicht erweitert werden
 
Oder der Methode noch die Direktive
Delphi-Quellcode:
stdcall;
mitgeben. Das sollte dann auch funktionieren.

Fritzew 23. Feb 2017 16:39

AW: Dynamisches Array in DLL kann nicht erweitert werden
 
Zitat:

Zitat von Aviator (Beitrag 1362447)
Oder der Methode noch die Direktive
Delphi-Quellcode:
stdcall;
mitgeben. Das sollte dann auch funktionieren.

Was hat das mit der stdcall (Aufrufkonvention) zu tun?
Das beeinflusst in keiner Weise das Speichermanagement, sondern nur wie übergebene Parameter auf dem Stack behandelt werden beim verlassen der Routine. Das hat nichts mit dem allozieren von Speicher zu tun. Ohne Sharemem wird das nicht funktionieren.
Damit dynamische Arrays und auch Strings mit Dll's funktionieren muss der gleiche Speichermanager benutzt werden

devidespe 27. Feb 2017 11:47

AW: Dynamisches Array in DLL kann nicht erweitert werden
 
Nachdem ich nun alle Vorschläge ausprobiert habe und Packages ausscheiden, half mir die Direktive:

Delphi-Quellcode:
stdcall;


innerhalb der DLL, und nun wird der OpenArray-Parameter korrekt in beide Richtungen übergeben.

ShareMem hatte ich auch versucht, allerdings gelang es auch ohne und soweit ich das verstanden habe, macht es nur Sinn, wenn die Applikation, welche später auf die DLL zugreift, auch eine Delphi-Applikation ist - was bei mir aber definitiv nicht der Fall ist.

Blup 6. Mär 2017 22:13

AW: Dynamisches Array in DLL kann nicht erweitert werden
 
Damit hast du den Fehler nicht beseitigt, nur das Symptom Zugriffsverletzung tritt zufällig an dieser Stelle nicht mehr auf. Wenn Speichermanager A Speicher freigibt, den Speichermanager B reserviert hat, wird Speichermanager B irgendwann versuchen darauf zuzugreifen. Das kann in einem völlig anderen Programmteil irgendwann wieder zur Zugriffsverletzung führen und du wirst nicht wissen warum.


Alle Zeitangaben in WEZ +1. Es ist jetzt 14:16 Uhr.
Seite 1 von 2  1 2      

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