文件流写入文件并读取bfile

时间:2010-06-08 17:31:31

标签: c# .net filestream

我有一个Web服务,它检查字典以查看文件是否存在,然后是否存在它会读取文件,否则会保存到文件中。这是来自网络应用程序。我想知道最好的方法是什么,因为如果同时访问同一个文件,偶尔会遇到FileNotFoundException异常。 这是代码的相关部分:

String signature;
signature = "FILE," + value1 + "," + value2 + "," + value3 + "," + value4;     // this is going to be the filename

string result;            
MultipleRecordset mrSummary = new MultipleRecordset();  // MultipleRecordset is an object that retrieves data from a sql server database

if (mrSummary.existsFile(signature))
{                
    result = mrSummary.retrieveFile(signature);                
}
else
{                
    result = mrSummary.getMultipleRecordsets(System.Configuration.ConfigurationManager.ConnectionStrings["MyConnectionString"].ConnectionString.ToString(), value1, value2, value3, value4);
    mrSummary.saveFile(signature, result);
}

以下是查看文件是否已存在的代码:

private static Dictionary<String, bool> dict = new Dictionary<String, bool>();

public bool existsFile(string signature)
{                       
    if (dict.ContainsKey(signature))
    {
        return true;
    }
    else
    {
        return false;
    }                                
}

以下是我用来检索它的内容:

try
{                               

    byte[] buffer;
    FileStream fileStream = new FileStream(@System.Configuration.ConfigurationManager.AppSettings["CACHEPATH"] + filename, FileMode.Open, FileAccess.Read, FileShare.Read);
      try
      {
        int length = 0x8000;  // get file length
        buffer = new byte[length];            // create buffer
        int count;                            // actual number of bytes read
        JSONstring = "";
        while ((count = fileStream.Read(buffer, 0, length)) > 0)
        {
            JSONstring += System.Text.ASCIIEncoding.ASCII.GetString(buffer, 0, count);
        }
      }
      finally
      {
        fileStream.Close();
      }
}
catch (Exception e)
{

    JSONstring = "{\"error\":\"" + e.ToString() + "\"}";                
}

如果该文件以前不存在,则会将JSON保存到文件中:

try 
{                
    if (dict.ContainsKey(filename) == false)
    {
        dict.Add(filename, true);
    }
    else
    {
        this.retrieveFile(filename, ipaddress);
    }

}
catch
{


}


try
{
    TextWriter tw = new StreamWriter(@System.Configuration.ConfigurationManager.AppSettings["CACHEPATH"] + filename);
    tw.WriteLine(JSONstring);
    tw.Close();
}
catch { 

}

以下是我运行上述代码时遇到的异常的详细信息:

System.IO.FileNotFoundException: Could not find file 'E:\inetpub\wwwroot\cache\FILE,36,36.25,14.5,14.75'.
File name: 'E:\inetpub\wwwroot\cache\FILE,36,36.25,14.5,14.75'
   at System.IO.__Error.WinIOError(Int32 errorCode, String maybeFullPath)
   at System.IO.FileStream.Init(String path, FileMode mode, FileAccess access, Int32 rights, Boolean useRights, FileShare share, Int32 bufferSize, FileOptions options, SECURITY_ATTRIBUTES secAttrs, String msgPath, Boolean bFromProxy)
   at System.IO.FileStream..ctor(String path, FileMode mode, FileAccess access, FileShare share)
   at com.myname.business.MultipleRecordset.retrieveFile(String filename, String ipaddress)

2 个答案:

答案 0 :(得分:3)

您得到FileNotFoundException,因为您在实际编写文件之前将文件名添加到字典中。因此,对同一文件的并发操作将导致此问题。

写完后将它添加到字典中只会产生一个新问题:它会开始尝试同时写入文件(并且失败了)。如果性能至关重要,我会将字典更改为Dictionary<string, object>,并将该值用作文件级别的同步对象。您还需要添加一个单独的同步对象,以便检查并添加到字典本身。

但这可能有点过分,需要手动使用Monitor.EnterMonitor.Exit。这是一个稍微简单的实现:

static HashSet<string> files = new HashSet<string>();
static object syncRoot = new object();

void Whatever(string filename, string ipaddress)
{
    bool fileFound;

    lock (syncRoot)
    {
        fileFound = files.Contains(filename);
        if (!fileFound)
        {
            files.Add(filename);
            // Code to write file here
        }
    }

    if (fileFound)
    {
        retrieveFile(filename, ipaddress);
    }
}

写入完成后,性能会有点次优,因为它会阻止所有读取操作,直到写入完成。如果大多数操作都是读操作,那么这不是问题。

在本例中,我已将字典更改为HashSet,因为您似乎只使用了密钥,而不是值。

答案 1 :(得分:1)

这是一个线程问题。

FileNotFoundException发生在这一行

  FileStream fileStream = new FileStream(System.Configuration.ConfigurationManager.AppSettings["CACHEPATH"] + filename, FileMode.Open, FileAccess.Read, FileShare.Read);  

因为该文件尚不存在。

你正在打电话

dict.Add(filename, true);

在创建文件之前

TextWriter tw = new StreamWriter(@System.Configuration.ConfigurationManager.AppSettings["CACHEPATH"] + filename); 
tw.WriteLine(JSONstring); 
tw.Close(); 

导致您对文件存在的测试不准确

public bool existsFile(string signature)       
{                              
    if (dict.ContainsKey(signature))       
    {       
        return true;       
    }       
    else       
    {       
        return false;       
    }                                       
}