在没有读取器客户端访问中断的情况下替换文件内容

时间:2011-05-21 15:37:58

标签: c# concurrency filesystems writer

有一项服务不断将新内容写入文件:

using (var stream = File.Create(FileName))     // overwrites the file
{
    stream.Write(data, 0, data.Length);
}

该文件经常被多个读者访问(包括从该文件呈现其内容的Web应用程序)。我无法控制读者客户代码。读者应始终可以访问该文件。更重要的是,他们应该看到整个内容,而不是写入文件的中间内容。

任何类似的技巧:

using (var stream = File.Create(FileName + ".tmp"))
{
    stream.Write(data, 0, data.Length);
}

File.Delete(FileName);
File.Move(FileName + ".tmp", FileName);

可能导致网页上缺少内容(有一定的可能性)。并且该服务有时会抛出IOException异常,并显示消息“进程无法访问该文件,因为它正被另一个进程使用”。

问题是:如果没有读者客户端访问中断,如何不断更换文件内容?

1 个答案:

答案 0 :(得分:0)

在IIS中,您可以调整this module(我写完全披露)以将同步注入到读取请求中。您可以通过继承InterceptingHandler并添加如下代码来完成此操作:

SychronizingHandler : InterceptingHandler
{
    // ...

    Semaphore mySemaphore;

    protected override bool PreFilter(System.Web.HttpContext context)
    {
        context.RewritePath("myFilePath");
        if( mySemaphore == null)
        {
            bool created;
            mySemaphore = new Semaphore(100, 0, "semphoreName", out created);
        }

        if( mySemaphore != null)
        {
            mySemaphore.WaitOne();
        }
        reutrn true;
    }

    // note this function isn't in the base class
    // you would need to add it  and call it right after the call to
    // innerHandler.ProcessRequest
    protected override void PostFilter(System.Web.HttpContext context) 
    {
        mySemaphore.Release();
        return;
    }

    protected virtual void OnError(HttpContext context, Exception except)
    {
        mySemaphore.Release();
        return base.OnError(context, except);
    }

桌面应用程序有点棘手,因为它取决于应用程序的实现细节。希望在这种情况下,您可以通过某种方式扩展它并添加同步。

正如Fun在评论中指出的那样,您也可以在预过滤器中进行条件重写,这样您就不会尝试访问正在编写的文件,这是一个非常好的主意。