我编写了一个简单的线程同步程序。但是当我运行这个程序时,我收到一个错误“进程无法访问文件'D:\ Vivek.txt',因为它正被另一个进程使用。”为什么我收到此错误。
class Program
{
const string Filepath = "D:\\Vivek.txt";
static AutoResetEvent writerwaithandle= new AutoResetEvent(true);// Signaled state
static AutoResetEvent readerwaithandle = new AutoResetEvent(false);
static void Main()
{
if (File.Exists(Filepath))
{
File.Delete(Filepath);
}
File.CreateText(Filepath);
CreateWriterThread();
CreateReaderThread();
Console.ReadKey();
}
private static void CreateWriterThread()
{
for (int i = 1; i <= 10;i++ )
{
var thread = new Thread(WriteFile);
thread.Name = "Writer " + i;
thread.Start();
Thread.Sleep(250);
}
}
private static void CreateReaderThread()
{
for (int i = 1; i <= 10; i++)
{
var thread = new Thread(ReadFile);
thread.Name = "Reader " + i;
thread.Start();
}
}
private static void WriteFile()
{
writerwaithandle.WaitOne();
var stream = new FileStream(Filepath, FileMode.Append);
var streamwriter = new StreamWriter(stream);
streamwriter.WriteLine("written by"+Thread.CurrentThread.Name+DateTime.Now));
streamwriter.Flush();
streamwriter.Close();
readerwaithandle.Set();
}
private static void ReadFile()
{
readerwaithandle.WaitOne();
if (File.Exists(Filepath))
{
var stream = new FileStream(Filepath, FileMode.Open);
var streamreader = new StreamReader(stream);
var text = streamreader.ReadToEnd();
streamreader.Close();
Console.WriteLine("Read by thread {0} \n",Thread.CurrentThread.Name);
Console.WriteLine(text);
}
writerwaithandle.Set();
}
}
当我从
替换代码时if (File.Exists(Filepath))
{
File.Delete(Filepath);
}
File.CreateText(Filepath);
到
if (!File.Exists(Filepath))
{
File.CreateText(Filepath);
}
程序第一次显示相同的错误。之后它永远不会给出任何错误。 请任何人告诉我bug区域,原因以及最佳解决方案。
答案 0 :(得分:2)
使用FileStream
时,请始终使用using
using (var fileStream = new FileStream(Filepath, FileMode.Open))
{
Your code...
}
这可以确保流正确处理。
您还可以使用StreamReader
和using
using (var fileStream = new FileStream(Filepath, FileMode.Open))
{
using (var reader = new StreamReader(stream))
{
Your code...
}
}
答案 1 :(得分:1)
警惕,look在File.CreateText
创建或打开用于编写UTF-8编码文本的文件。
返回值类型:System.IO.StreamWriter写入
的StreamWriter使用UTF-8编码
指定文件
这意味着无需创建新的FileStream,因为在使用File.CreateText
时已经创建并返回了FileStream。您应该只在代码中使用创建的FileStream。
以下是您的代码的固定版本:
class Program
{
const string Filepath = "D:\\Vivek.txt";
static AutoResetEvent writerwaithandle = new AutoResetEvent(true);// Signaled state
static AutoResetEvent readerwaithandle = new AutoResetEvent(false);
static void Main()
{
if (File.Exists(Filepath))
{
File.Delete(Filepath);
}
//File.CreateText(Filepath);
CreateWriterThread();
CreateReaderThread();
Console.ReadKey();
}
private static void CreateWriterThread()
{
for (int i = 1; i <= 10; i++)
{
var thread = new Thread(WriteFile);
thread.Name = "Writer " + i;
thread.Start();
Thread.Sleep(250);
}
}
private static void CreateReaderThread()
{
for (int i = 1; i <= 10; i++)
{
var thread = new Thread(ReadFile);
thread.Name = "Reader " + i;
thread.Start();
}
}
private static void WriteFile()
{
writerwaithandle.WaitOne();
var streamwriter = File.CreateText(Filepath);
//var stream = new FileStream(Filepath, FileMode.Append);
//var streamwriter = new StreamWriter(stream);
streamwriter.WriteLine("written by" + Thread.CurrentThread.Name + DateTime.Now);
streamwriter.Flush();
streamwriter.Close();
readerwaithandle.Set();
}
private static void ReadFile()
{
readerwaithandle.WaitOne();
if (File.Exists(Filepath))
{
var stream = new FileStream(Filepath, FileMode.Open);
var streamreader = new StreamReader(stream);
var text = streamreader.ReadToEnd();
streamreader.Close();
Console.WriteLine("Read by thread {0} \n", Thread.CurrentThread.Name);
Console.WriteLine(text);
}
writerwaithandle.Set();
}
}