Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Programmieren allgemein (https://www.delphipraxis.net/40-programmieren-allgemein/)
-   -   mal wieder Verzeichnis kopieren (https://www.delphipraxis.net/155204-mal-wieder-verzeichnis-kopieren.html)

bwolf 13. Okt 2010 07:59

mal wieder Verzeichnis kopieren
 
Hi Leute,

das Thema gibt es zwar schon 50x, muß aber trotzdem nochmal fragen...
Ich in einer Anwendung ein Verzeichnis kopieren.
Dazu findet man ja überall folgenden Code:

Code:
function CopyDir(const fromDir, toDir: string): Boolean;
var
  fos: TSHFileOpStruct;
begin
  ZeroMemory(@fos, SizeOf(fos));
  with fos do
  begin
    wFunc := FO_COPY;
    fFlags := FOF_FILESONLY;
    pFrom := PChar(fromDir + #0);
    pTo   := PChar(toDir)
  end;
  Result := (0 = ShFileOperation(fos));
end;
Unter Win7, Vista und Server 2008 habe ich mit dieser Funktion kein Problem.
Unter Windows XP kriege ich allerdings folgenden Fehler:

---------------------------
Fehler beim Kopieren der Datei oder des Ordners
---------------------------
Datei kann nicht kopiert werden: Die Quelldatei oder vom Quelldatenträger kann nicht gelesen werden.
---------------------------
OK
---------------------------

Hat einer eine Idee was da faul ist?
Danke!

sx2008 13. Okt 2010 09:11

AW: mal wieder Verzeichnis kopieren
 
Delphi-Quellcode:
// der Fehler liegt hier
pFrom := PChar(fromDir + #0);
Was passiert da?
Delphi erzeugt einen neuen temporären String mit dem Inhalt fromDir+#0.
Dieser temporäre String wird ungültig ab der nächsten Zeile.
Der Zeiger pFrom zeigt also auf ungültige Daten.
Zu guter Letzt kommt der Aufruf von ShFileOperation.

Jetzt hängt es vom Zufall ab, bzw. von Dingen, die man als Programmierer nicht beeinflussen kann,
ob der Speicherblock auf den der Zeiger verweist noch intakt ist oder nicht.

Die Lösung des Problems ist eine Zwischenvariable:
Delphi-Quellcode:
var
  fromDir_mit_null : string;
....
  fromDir_mit_null := fromDir + #0;
  pFrom := PChar(fromDir_mit_null);
Falls du den defekten Code hier von der DP kopiert hast, bitte melden,
damit nicht noch mehr Leute darauf reinfalllen.

bwolf 13. Okt 2010 09:17

AW: mal wieder Verzeichnis kopieren
 
Hi,
Problem war, dass das Quellverzeichnis ein "\" am Ende hatte - dies hatte unter Win7 und CO funktioniert, aber unter windows xp die besagte Fehlermeldung hervorgerufen.

DeddyH 13. Okt 2010 09:20

AW: mal wieder Verzeichnis kopieren
 
Müssten es nicht 2 #0 am jeweiligen Ende sein? http://msdn.microsoft.com/en-us/libr...=VS.85%29.aspx
Zitat:

pFrom
LPCTSTR

Note This string must be double-null terminated.
...
pTo
LPCTSTR

Note This string must be double-null terminated.
Ansonsten gibt es in der CodeLib auch eine Funktion von sakura, damit sollte es klappen.

sx2008 13. Okt 2010 09:31

AW: mal wieder Verzeichnis kopieren
 
Zitat:

Zitat von DeddyH (Beitrag 1055480)
Müssten es nicht 2 #0 am jeweiligen Ende sein?

Da ein (Ansi-)String in Delphi schon implizit eine #0 am Ende hat, sind dann effektiv zwei Nullen im Speicher.
Vielleicht wäre es besser wenn man im Code +#0#0 schreiben würde um diese Unklarheit zu beseitigen...

himitsu 13. Okt 2010 09:49

AW: mal wieder Verzeichnis kopieren
 
Zitat:

Delphi-Quellcode:
pFrom := PChar(fromDir + #0);

Delphi erzeugt hier eine temporäre Variable vom selben Typ wie fromDir (der erste Operand) und mit dem Inhalt von
Delphi-Quellcode:
fromDir + #0
,
übergibt davon den Zeiger an pFrom und zum Ende der Funktion wird diese Variable freigegeben.

In dieser Variable steht also
Delphi-Quellcode:
fromDir + #0
als Text und daran anknüpfend noch ein zustätliches #0, welches der Delphisteing enthält, um kompatibel zum PChar zu sein.
Die #0 im Text und die #0 des String ergeben also #0#0. :angel:

Fazit, es funktioniert (solange man es nicht unbedingt innerhalb einer Schleife nutzt) und wird auch (von mir) seit Jahren erfolgreich eingesetzt.


Alle Zeitangaben in WEZ +1. Es ist jetzt 07:16 Uhr.

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