Search code examples
c#file-iotextreadertextwriter

Sharing violation IOException while reading and writing to file C#


Here is my code:

public static TextWriter twLog = null;
private int fileNo = 1;
private string line = null;

TextReader tr = new StreamReader("file_no.txt");
TextWriter tw = new StreamWriter("file_no.txt");
line = tr.ReadLine();
if(line != null){
    fileNo = int.Parse(line);
    twLog = new StreamWriter("log_" + line + ".txt");
}else{
    twLog = new StreamWriter("log_" + fileNo.toString() + ".txt");  
}
System.IO.File.WriteAllText("file_no.txt",string.Empty);
tw.WriteLine((fileNo++).ToString());
tr.Close();
tw.Close();
twLog.Close();

It throws this error:

IOException: Sharing violation on path C:\Users\Water Simulation\file_no.txt

What i'm trying to do is just open a file with log_x.txt name and take the "x" from file_no.txt file.If file_no.txt file is empty make log file's name log_1.txt and write "fileNo + 1" to file_no.txt.After a new program starts the new log file name must be log_2.txt.But i'm getting this error and i couldn't understand what am i doing wrong.Thanks for help.


Solution

  • Well, you're trying to open the file file_no.txt for reading and for writing using separate streams. This may not work as the file will be locked by the reading stream, so the writing stream can't be created and you get the exception.

    One solution would be to read the file first, close the stream and then write the file after increasing the fileNo. That way the file is only opened once at a time.

    Another way would be to create a file stream for both read and write access like that:

    FileStream fileStream = new FileStream(@"file_no.txt", 
                                           FileMode.OpenOrCreate, 
                                           FileAccess.ReadWrite, 
                                           FileShare.None);
    

    The accepted answer to this question seems to have a good solution also, even though I assume you do not want to allow shared reads.

    Possible alternate solution
    I understand you want to create unique log files when your program starts. Another way to do so would be this:

    int logFileNo = 1;
    string fileName = String.Format("log_{0}.txt", logFileNo);
    
    while (File.Exists(fileName))
    {
        logFileNo++;
        fileName = String.Format("log_{0}.txt", logFileNo);
    }
    

    This increases the number until it finds a file number where the log file doesn't exist. Drawback: If you have log_1.txt and log_5.txt, the next file won't be log_6.txt but log_2.txt.

    To overcome this, you could enumerate all the files in your directory with mask log_*.txt and find the greatest number by performing some string manipulation.

    The possibilities are endless :-D