Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

FileStream locking a file for reading and writing

I have the following code block which is giving me a headache.

Logically it should work as I am using the filestream providing the lock within the using statement. When it gets to the line that creates the StreamWriter, it fails saying "the file is not writable".

Now my program is a multithreaded application. Any thread could be trying to write to this file. I need the program to lock the file, read the contents, check the contents, then write back any changes. During that process, no other thread should be able to access that file.

using (var fs = File.Open(fileLoc, FileMode.Open, FileAccess.ReadWrite, FileShare.None))
{
    var sr = new StreamReader(fs);
    var str = sr.ReadToEnd();
    var strArray = str.Split(',');
    if (strArray.Any(st => st == text))
    {
        return;
    }
    sr.Close();

    var sw = new StreamWriter(fs);
    sw.Write(str + text);
    sw.Flush();
    sw.Close();
}
like image 327
Talon Avatar asked Dec 04 '12 07:12

Talon


1 Answers

The FileShare.None flag does not cause threads to queue, it just locks the file, hence the exception that you get. To provide mutually exclusive access you can lock a shared object prior to writing.

But you say this "Now my program is a multithreaded application. Any thread could be trying to write to this file." Now, do these threads all use exactly the same method to write to the file? Let's assume they do, then this should work ...

Create a static class variable ...

private static object lockObject = new object();

Use it here ...

lock (lockObject) 
{
    using(var sw = new StreamWriter(fs))
    {
       sw.Write(str + text);
    }
}

I have made some assumptions about the threads, so you may have to look up info on synchronization if this doesn't work or provide some more info to us.

Also, please close your StreamReader earlier (in case the method returns earlier). Close it immediately after you use it or better yet use using.

like image 117
xagyg Avatar answered Sep 18 '22 15:09

xagyg