![]() |
Verzeichnisüberwachung
Moin,
ich habe mir mal ein Programm geschrieben was ein vom benutzer eingestelltes Verzeichnis überwacht und bei Änderungen darin überprüft ob alle Dateien darin i.O. sind. Das wird mit Hilfe eines Threads gemacht.
Delphi-Quellcode:
Jetzt habe ich mal eine neuen Datei angelegt und wieder gelöscht dann ist das alles kein Problem. Die Änderungen wird mit AddFileToList in eine TStringList eingefügt und kann bei bedarf angezeigt ode gespeichert werden. Anschließend habe ich mal mit TotalCommander das Dateidatum geändert und wollte mir dann mal die StringList angucken. Da trat mein Problem auf: In meiner StringList waren fast 500000 einträge. Das Speichern der Datei war möglich, dauerte natürlich sehr lange und die Datei war anchher 142 MB groß (reine Textdatei), aber das Anzeigen der StringList in TListBox mit Assign ging gar nicht mehr. Nachdem die Auslagerungsdatei auf 1,94 GB erweitert wurde, wollte mein Rechner nicht mehr.
procedure TcsDirThread.Execute;
var pBuf : Pointer; dwBufLen : DWORD; dwRead : DWORD; FNI : FILE_NOTIFY_INFORMATION; pWork : Pointer; sFileName : Widestring; dwWaitStatus : DWORD; begin FhFile := CreateFile(PChar(FsDirPath), FILE_LIST_DIRECTORY or GENERIC_READ, FILE_SHARE_READ or FILE_SHARE_WRITE or FILE_SHARE_DELETE, nil, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS,0); hNotifity := FindFirstChangeNotification(PChar(FsDirPath), //Verzeichnis false, //unterverzeichnisse überwachen FILE_NOTIFY_CHANGE_FILE_NAME or FILE_NOTIFY_CHANGE_LAST_WRITE or FILE_NOTIFY_CHANGE_SIZE or FILE_ACTION_ADDED or FILE_ACTION_REMOVED or FILE_ACTION_MODIFIED); if (FhFile = INVALID_HANDLE_VALUE) or (FhFile = 0) then begin RaiseLastWin32Error; Terminate; end; if (hNotifity = INVALID_HANDLE_VALUE) then begin RaiseLastWin32Error; Terminate; end; dwBufLen := 65536; pBuf := AllocMem(dwBufLen); try while ((FindNextChangeNotification(hNotifity)) and (not terminated)) do begin Synchronize(Application.ProcessMessages); dwWaitStatus := WaitForSingleObject(hNotifity, 1000); if (dwWaitStatus = WAIT_FAILED) then begin RaiseLastWin32Error; Terminate; end; if (dwWaitStatus = WAIT_OBJECT_0) then begin ReadDirectoryChangesW(FhFile,pBuf,dwBufLen,true, FILE_NOTIFY_CHANGE_FILE_NAME or FILE_NOTIFY_CHANGE_DIR_NAME or FILE_NOTIFY_CHANGE_ATTRIBUTES or FILE_NOTIFY_CHANGE_SIZE or FILE_NOTIFY_CHANGE_LAST_WRITE or FILE_NOTIFY_CHANGE_CREATION or FILE_ACTION_ADDED or FILE_ACTION_REMOVED or FILE_ACTION_MODIFIED, @dwRead,nil,nil); pWork := pBuf; repeat StrMove(@FNI,pWork,12); PChar(pWork) := PChar(pWork)+12; sFileName := StringOfChar(#00,FNI.dwFileNameLength); StrMove(@sFileName[1],pWork,FNI.dwFileNameLength); FsFileName := WideCharToString(PWideChar(sFileName)); FsFileName := copy(FsFileName,1,length(FsFileName) shl 1); FsReason := GetReason(FNI.dwAction); Synchronize(AddFileToList); PChar(pWork) := PChar(pBuf)+FNI.dwNextEntryOffset; until FNI.dwNextEntryOffset = 0; end; end; if (LowerCase(ExtractFileExt(FsFileName)) = '.bmp') then begin PostMessage(MainForm.Handle, Verzeichnisaenderung_bmp, 0, ProjektID); end; finally FreeMem(pBuf,dwBufLen); end; end; Warum sind da sol viele Änderungen drin. Das lustige dabei ist das bis auf die ersten 2 Änderungen (Datei anlegen und löschen) immer das selbst stand (Datei wurde geändert). Immer wieder das selbe. Im Abstand von weniger als 1 sekunde. Ist das jetzt ein Problem bei mir oder ein Problem von Windows das ich umgehen muss? Wäre nett wenn ihr mir Tipps und Anregungen geben würdet. |
Re: Verzeichnisüberwachung
Was genau der Total Commander macht, ist fraglich ... was hast du denn unternommen um das Dateidatum zu ändern? Klingt für mich danach, als würde "einfach" die Datei kopiert oder neu angelegt o.ä.
Es gibt zB auch Editoren die von der zu bearbeitenden Datei eine Kopie anlegen. Das kann durchaus ein Problem sein im Falle von Hardlinks! Denn dann wird der Hardlink zerbrochen und "die Datei" existiert wieder als "zwei verschiedene Dateien". An deiner Stelle würde ich den Code so ändern, daß ich alle Aktionen auf eine Datei dann eben zusammenfasse, bis sich der Dateiname ändert ... sozusagen "stateful" machen. Ansonsten wirst du wohl damit leben müssen. Du kannst ja mal gern den FileMon von Sysinternals testen, aber das Ergebnis wird sehr ähnlich sein. |
Re: Verzeichnisüberwachung
Ich habe die Datei in TotalCommander markiert bin dann im Menü "Datei" auf "Dateiattribute ändern" gegangen und habe dort "Datum/Zeit" auf die aktuelle Zeit gesetzt.
Ich könnte zwar alle Ergebnisse dieser einen Datei zu einem zusammenfassen, aber es kann ja sein das sich die Datei heute ändert und morgen auch noch mal. Dazu ist das Protokoll ja da, um das zu protokollieren. Das Programm soll auf einem Server laufen und alle Dateien überprüfen ob die noch korrekt sind, wenn eine Änderung drinne war. Und fast 500000 mal den Test starten ob die Dateien noch korrekt sind ist auch doof, das bringt den Rechner an den Rand des Absturtzes. Vor allem da nur eine Änderung gemacht wurde. |
Re: Verzeichnisüberwachung
Aber "noch korrekt" bezieht sich doch sicherlich auf den Inhalt. Überlege mal was du dort alles für Änderungen überwachst, das allein ist doch schon irre auf einem Server! Stattdessen sollte sich der Thread schlafenlegen (zB 10s), wenn er eine Änderung bemerkt hat und entsprechende Maßnahmen getroffen hat. Außerdem sollten nur relevante Änderungen überwacht und dann vorzugsweise nochmal von dir gefiltert werden.
|
Re: Verzeichnisüberwachung
Kann ich einen Thread auch einfach mit Sleep pausieren?
Ich dachte das wäre so die wichtigsten Änderungen: FILE_NOTIFY_CHANGE_FILE_NAME - Datiename hat sich geändert FILE_NOTIFY_CHANGE_DIR_NAME - Verzeichnisname hat sich geändert FILE_NOTIFY_CHANGE_ATTRIBUTES - Dateiattribut hat sich geändert FILE_NOTIFY_CHANGE_SIZE - Dateigröße hat sich geändert FILE_NOTIFY_CHANGE_LAST_WRITE - Datei gespeichert FILE_NOTIFY_CHANGE_CREATION - Datei ersteltl FILE_ACTION_ADDED - Date ist hinzugekommen FILE_ACTION_REMOVED - Datei wurde gelöscht FILE_ACTION_MODIFIED - Datei wurde geändert Ok, ich denke das Größe, Attribut ändern brauch ich nicht zu überwachen und FILE_NOTIFY_CHANGE_CREATION werde ich auch rausnehmen. Der Rest erscheint mir eigentlich sehr sinnvoll. |
Re: Verzeichnisüberwachung
Zitat:
Zitat:
|
Re: Verzeichnisüberwachung
Mein Ziel:
Auf einem Server werden in unregelmäßigen Abständen Dateien kopiert oder ersetzt. Jetzt soll nach jeder Änderung überprüft werden ob die Dateien noch i.O. sind. Das Überprüfen ist kein Problem das funzt soweit auch vollständig. Bloß diese 500000 Ändeurngen am Verzeichnis machen mir Porbleme. |
Re: Verzeichnisüberwachung
Okay, wodurch bestimmt sich "i. O."??? Durch den Inhalt? Änderungszeit? Attribute? Größe?
|
Re: Verzeichnisüberwachung
Das wird anhand des Inhaltes festegelegt, das heißt in welchen speziellen Format die Dateien vorliegen (bei Bmp's farbtiefe, etc.)
|
Re: Verzeichnisüberwachung
Okay, dann gehen wir mal die Flags durch:
Sobald eine Änderung passiert, solltest du den Thread schlafenlegen nachdem du deine Überprüfung gemacht hast. Nach dem Aufwachen machst du den nächsten Durchgang. Allerdings verstehe ich noch nicht, wieso du FindNextChangeNotification und ReadDirectoryChangesW mischst. IMO reicht es in einem Thread, wenn du einfach die synchrone Variante von ReadDirectoryChangesW aufrufst und somit die Funktion erst zurückkehrt, wenn was passiert ist. Danach kann sich der Thread wieder schlafenlegen indem die Funktion wieder aufgerufen wird (ich empfehle davor ein Sleep(0);). Die beiden Methoden zur Ermittlung von Änderungen müssen nicht kombiniert eingesetzt werden! FindFirst*/Next* ermittelt nur das OB, ReadDirectory* ermittelt das WAS! Das ist der Unterschied. Ich empfehle letztere Methode, da dies die für dich passende zu sein scheint. |
Alle Zeitangaben in WEZ +1. Es ist jetzt 05: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