Наблюдатель за файловой системой и большие файлы - PullRequest
12 голосов
/ 29 сентября 2010
var fsw = new FileSystemWatcher(sPath, "*.PPF");
fsw.NotifyFilter = NotifyFilters.FileName;
fsw.IncludeSubdirectories = true;
fsw.Created += FswCreated;
fsw.EnableRaisingEvents = true;

static void FswCreated(object sender, FileSystemEventArgs e)
{
  string sFile = e.FullPath;
  string[] arrLines = File.ReadAllLines(sFile);
}

это не работает с большими файлами, потому что процесс не заканчивается записью файла. Файл копируется через сеть, поэтому я не знаю его размера. Какой тип синхронизации требуется, чтобы сделать это устойчивым?

Ответы [ 5 ]

28 голосов
/ 29 сентября 2010

Решение найдено в stackoverflow и немного его изменено.

static bool IsFileLocked(FileInfo file)
{
    FileStream stream = null;

    try
    {
        stream = file.Open(FileMode.Open, 
                 FileAccess.ReadWrite, FileShare.None);
    }
    catch (IOException)
    {
        //the file is unavailable because it is:
        //still being written to
        //or being processed by another thread
        //or does not exist (has already been processed)
        return true;
    }
    finally
    {
        if (stream != null)
            stream.Close();
    }

    //file is not locked
    return false;
}

static void FswCreated(object sender, FileSystemEventArgs e)
{
    string sFile = e.FullPath;

    Console.WriteLine("processing file : " + sFile);

    // Wait if file is still open
    FileInfo fileInfo = new FileInfo(sFile);
    while(IsFileLocked(fileInfo))
    {
        Thread.Sleep(500);
    }

    string[] arrLines = File.ReadAllLines(sFile);
}
4 голосов
/ 29 сентября 2010

Используйте DelayedFileSystemWatcher.cs Класс http://blogs.msdn.com/b/ahamza/archive/2006/02/06/526222.aspx

а затем этот код. Проверьте обработчик событий PrintFileSystemEventHandler. Он пытается прочитать файл в файловом потоке, и, если возникает какой-либо ioerror, он предполагает, что файл все еще читает, поэтому он ожидает некоторый интервал (в этом примере 2 секунды) и затем пытается снова. Проверьте CONVERSION: метку

static void Main(string[] args)
    {
        DelayedFileSystemWatcher dfw = new DelayedFileSystemWatcher(@"C:\\Test", "*.*");
        dfw.Changed += new FileSystemEventHandler(Program.FileSystemEventHandlerMethod);
        dfw.Created += new FileSystemEventHandler(Program.FileSystemEventHandlerMethod);
        dfw.Deleted += new FileSystemEventHandler(Program.FileSystemEventHandlerMethod);
        dfw.Error += new ErrorEventHandler(Program.ErrorEventHandlerMethod);
        dfw.Renamed += new RenamedEventHandler(Program.RenamedEventHandlerMethod);

        dfw.IncludeSubdirectories = true;
        dfw.ConsolidationInterval = 1000;
        dfw.EnableRaisingEvents = true;
        Console.WriteLine("Press \'q\' to quit the sample.");
        while (Console.Read() != 'q') ;
        //System.Threading.Thread.Sleep(60000);
        dfw.Dispose();
    }
    private static void FileSystemEventHandlerMethod(object sender, FileSystemEventArgs e)
    {
        PrintFileSystemEventHandler(e);
        System.Console.WriteLine();
    }

    private static void ErrorEventHandlerMethod(object sender, ErrorEventArgs e)
    {
        System.Console.WriteLine(e.GetException().Message);
        System.Console.WriteLine();
    }

    private static void RenamedEventHandlerMethod(object sender, RenamedEventArgs e)
    {
        PrintRenamedEventHandler(e);
        System.Console.WriteLine();
    }

    private static void PrintFileSystemEventHandler(FileSystemEventArgs e)
    {

    CONVERSION:
        try
        {

            if (e.ChangeType != WatcherChangeTypes.Deleted)
            {
                if (!isFolder(e.FullPath) && isFile(e.FullPath))
                {
                    FileStream fs = new FileStream(e.FullPath, FileMode.Open, FileAccess.Read, FileShare.None);
                    fs.Close();
                }

            }
            System.Console.WriteLine(e.Name + " " + e.FullPath + " " + e.ChangeType);

        }
        catch (System.IO.IOException)
        {
            Console.WriteLine("There was an IOException error or File is still copying. Retrying in 2 seconds...");
            System.Threading.Thread.Sleep(2000);
            goto CONVERSION;
        }



        //System.Console.WriteLine(e.Name);
        //System.Console.WriteLine(e.FullPath);
        //System.Console.WriteLine(e.ChangeType);
    }

    private static bool isFolder(string strPath)
    {

        bool isFolderExist = false;
        try
        {
            isFolderExist = Directory.Exists(strPath);
        }
        catch
        {
            isFolderExist = false;
        }
        return isFolderExist;
    }

    private static bool isFile(string strPath)
    {

        bool isFileExist = false;
        try
        {
            isFileExist = File.Exists(strPath);
        }
        catch
        {
            isFileExist = false;
        }
        return isFileExist;
    }


    private static void PrintRenamedEventHandler(RenamedEventArgs e)
    {
        PrintFileSystemEventHandler(e);
        System.Console.WriteLine(e.OldName);
        System.Console.WriteLine(e.OldFullPath);
    }

У меня нет ссылки на проект. но это поможет.

2 голосов
/ 31 января 2014

Я поддерживаю решение, принятое Шей Эрличмен.Но, тем не менее, а) Вы можете не открывать Файл с режимом доступа FileAccess.Read, если его файл доступен только для чтения

b) Некоторые программы при загрузке файла будут иметь забавное расширение и по завершениирасширение изменится, и, хотя файл будет завершен, у вас будет исключение файла не найдено.

Поэтому обработайте исключения, а также подпишитесь на событие file.renamed

1 голос
/ 29 сентября 2010

AFAIK вы не получите уведомление, как только копия будет готова, вы можете реализовать механизм повторных попыток.
Если вы получили нарушение сдвига, просто активируйте таймер, чтобы повторить операцию через X секунд.
Вторая повторная попытка должна быть через X * 2 секунды и т. Д. (С некоторыми ограничениями, конечно).

0 голосов
/ 29 сентября 2010

Просто в вашем fswCreated, поспите около 1/2 секунды с Thread.Sleep(500), если это возможно. Это должно дать вам время, необходимое компьютеру для завершения записи файла.

Конечно, для более медленных жестких дисков это может или может быть достаточно времени.

...