Autor Beitrag
CB
ontopic starontopic starontopic starontopic starontopic starontopic starontopic starofftopic star
Beiträge: 23



BeitragVerfasst: Mi 19.03.03 08:38 
Hallo! :)

Ich habe ein paar merkwürdige Probleme mit dynamischen Arrays vom Typ PChar.

Ich überge an eine Funktion in einer DLL ein dynamisches Array vom Typ PChar mit Parametern:

ausblenden Quelltext
1:
2:
3:
4:
5:
6:
TDynArray = array of PChar;

function SetParameters(pParameters: TDynArray): Boolean; cdecl; begin
  Parameters[0] := pParameters;
  result := True;
end;


Parameters ist ein globales Array in der DLL, was wie folgt deklariert ist:

ausblenden Quelltext
1:
2:
TParameters = array [0..1] of TDynArray;
Parameters: TParameters;


In Feld 0 werden die Parameter gespeichert; in Feld 1 die entsprechenden Beschreibungen zu den Parametern.
Wenn ich allerdings nun die Funktion SetParameters mit einem Array von Parametern aufrufe und mir danach die neuen Parameter per

ausblenden Quelltext
1:
2:
3:
4:
function GetParameters: TParameters; cdecl;
begin
  result := Parameters;
end;


zurückgeben lasse, dann scheint immer nur der erste Buchstabe eines Parameters gespeichert zu werden, da result[0][0] = 'b' ist, obwohl es per SetParameters eigentlich auf 'bla' gesetzt worden ist.

Zusammenfassung:
-----------------------
Deklaration:
ausblenden Quelltext
1:
2:
TDynArray = array of PChar;
TParameters = array [0..1] of TDynArray; // 0 -> parameter; 1 -> beschreibungen


DLL:
ausblenden Quelltext
1:
2:
3:
4:
5:
6:
7:
8:
9:
10:
11:
Parameters: TParameters;

function SetParameters(pParameters: TDynArray): Boolean; cdecl; begin
  Parameters[0] := pParameters;
  result := True;
end;

function GetParameters: TParameters; cdecl;
begin
  result := Parameters;
end;


Main program:
ausblenden Quelltext
1:
2:
3:
4:
5:
6:
7:
8:
9:
10:
11:
12:
13:
pParameters: TDynArray;
Parameters: TParameters;

// set parameter - in this example only one
EditParams.Text := 'bla';
SetLength(pParameters, 1);
pParameters[0] := PChar(EditParams.Text);
SetParameters(pParameters);

// get parameter
Parameters := GetParameters;
EditParams.Text := Parameters[0][0];
// ->> EditParams.Text = 'b';

------------------------------

Wenn ich im begin/end Teil der DLL Parameters[0][0] := 'test' setze, dann bekomme ich auch 'test' von GetParameters zurückgegeben. Es liegt also wahrscheinlich an der SetParameters Funktion.

Ich hoffe jemand versteht mein Problem und kennt evtl. eine Lösung. Bin über jede Hilfe dankbar. :)

Moderiert von user profile iconKlabautermann: Code-Tags hizugefügt.

_________________
mfg\CB
UGrohne
ontopic starontopic starontopic starontopic starontopic starontopic starofftopic starofftopic star
Veteran
Beiträge: 5502
Erhaltene Danke: 220

Windows 8 , Server 2012
D7 Pro, VS.NET 2012 (C#)
BeitragVerfasst: Mi 19.03.03 08:49 
Also ich bin mir nicht sicher, ob das Dein Problem lösen würde, aber ich würde schonmal die Deklaration vereinfachen.

Ich würde einen Record vorziehen. Also etwa so

ausblenden Quelltext
1:
2:
3:
4:
5:
6:
type TDynArray = record
       Parameter:PChar;
       Beschreibung:PChar;
       end;

var DynArray=array of PChar;


Du hast einen vereinfachten Zugriff und musst nicht 2 Arrays vergrößern.

Vielleicht löst das auch Dein Problem.
CB Threadstarter
ontopic starontopic starontopic starontopic starontopic starontopic starontopic starofftopic star
Beiträge: 23



BeitragVerfasst: Mi 19.03.03 10:31 
Das macht keinen Unterschied vom Problem her.
Das mit den zwei Arrays ist extra so, da man sonst an die SetParameter Funktion auch die Beschreibungen der Parameter übergibt, was nicht nötig ist, da die in der DLL definiert sind.

_________________
mfg\CB
CB Threadstarter
ontopic starontopic starontopic starontopic starontopic starontopic starontopic starofftopic star
Beiträge: 23



BeitragVerfasst: Do 20.03.03 14:51 
Also...hab die SetParameters Funktion jetzt so angeändert:

ausblenden Quelltext
1:
2:
3:
4:
5:
6:
7:
8:
9:
TDynArray = array of PChar;
TParameters = array [0..1] of TDynArray;

function SetNeededParameters(pParameters: TDynArray): Boolean; cdecl; Begin
  SetLength(Parameters[0], 1);
  GetMem(Parameters[0][0], StrLen(pParameters[0]) + 1);
  StrCopy(Parameters[0][0], pParameters[0]);
  result := True;
end;


Das funktioniert einwandfrei.

Allerdings gibts Probleme beim Aufrufen der Funktion. Hab das so probiert:

ausblenden Quelltext
1:
2:
3:
4:
5:
6:
TDynArray = array of PChar;
Parameters: TDynArray;

SetLength(Parameters, 1);
GetMem(Parameters[0], Length(EditSetParams.Text) + 1); StrCopy(Parameters[0], PChar(EditSetParams.Text));
SetParameters(Parameters);


Wenn man das so macht, dann wird allerdings nicht der komplette Inhalt von EditSetParams.Text in Parameters[0] gespeichert, sondern nur der erste Buchstabe davon.
Das wundert mich sehr, da ich folgendes Beispiel für die StrCopy Funktion in der Online Hilfe von Delphi gefunden habe:

ausblenden Quelltext
1:
2:
3:
4:
5:
6:
7:
8:
9:
10:
11:
procedure TFormMain.Button1Click(Sender: TObject);
var
  Buffer: PChar;
begin
  GetMem(Buffer,Length(Label1.Caption) + Length(Edit1.Text) + 1);
  StrCopy(Buffer, PChar(Label1.Caption));
  StrCat(Buffer, PChar(Edit1.Text));
  Label1.Caption := Buffer;
  Edit1.Clear;
  FreeMem(Buffer);
end;


Laut Beschreibung der Online Hilfe soll nach dem Klicken auf Button1 die Inhalte von Label1 und Edit1 kombiniert und dann in Label1 ausgeben werden. Hab das Beispiel per Copy&Paste ausprobiert und habe das gleiche Problem, wie mit dem Aufruf meiner SetParameters Funktion: Es wird nur jeweils der erste Buchstabe kopiert.

Weiß jemand, ob das ein Bug in Delphi oder ein Fehler in der Online Hilfe ist?
Hat jemand eine Lösung?

_________________
mfg\CB
CB Threadstarter
ontopic starontopic starontopic starontopic starontopic starontopic starontopic starofftopic star
Beiträge: 23



BeitragVerfasst: Fr 21.03.03 10:15 
Hi,

habs inzwischen selbst gelöst. Falls es jemanden interessiert:

Allg. Typdeklaration
ausblenden Quelltext
1:
2:
TDynArray = array of PChar;
TParameters = array [0..1] of TDynArray;


DLL:
ausblenden Quelltext
1:
2:
3:
4:
5:
6:
7:
8:
9:
10:
11:
12:
13:
14:
Parameters: TParameters;

function SetParameters(pParameters: TDynArray): Boolean; cdecl;
var
  a: Integer;
  
begin
  for a := 0 to Length(pParameters) - 1 do begin
    Parameters[0][a] := StrAlloc(StrLen(pParameters[a]) + 1);
    StrCopy(Parameters[0][a], pParameters[a]);
  end;

  result := True;
end;


Hauptprogramm
ausblenden Quelltext
1:
2:
3:
4:
5:
6:
7:
8:
9:
10:
11:
12:
Parameters: TDynArray;

SetLength(Parameters, 1);
Parameters[0] := StrAlloc(Length(EditSetParams.Text) + 1);
StrPCopy(Parameters[0], EditSetParams.Text + #0);

if not SetNeededParameters(Parameters) then
  ShowMessage('could not set parameters');

for a := 0 to Length(Parameters) - 1 do begin
  StrDispose(Parameters[a]);
end;

_________________
mfg\CB
MathiasSimmack
Ehemaliges Mitglied
Erhaltene Danke: 1



BeitragVerfasst: Fr 21.03.03 12:48 
*räusper* :lol: