如何避免异常进程无法访问该文件,因为它正由另一个进程使用。在.net

时间:2016-11-23 00:27:29

标签: c# .net file exception task-parallel-library

我使用以下代码解压缩a.gz文件。由于我有大量文件,我使用TPL任务来运行此代码。但我曾经得到.NET例外:

  

该进程无法访问该文件,因为该文件正由另一个进程使用。

如何解决此问题?
使用的代码如下:

private static void Decompress(FileInfo fileToDecompress)
{
    using (FileStream originalFileStream = fileToDecompress.OpenRead())
    {
        string currentFileName = fileToDecompress.FullName;
        string newFileName = currentFileName.Remove(currentFileName.Length - 
                                          fileToDecompress.Extension.Length);
        using (FileStream decompressedFileStream = File.Create(newFileName))
        {
            using (GZipStream decompressionStream = 
                     new GZipStream(originalFileStream, 
                                    CompressionMode.Decompress))
            {
                decompressionStream.CopyTo(decompressedFileStream);
            }
        }
    }
    //File.Move(myffile, Path.ChangeExtension(myffile, ".jpg"));
}

调用代码如下所示

var list = ftp.GetFileList(remotepath);

//-------------------
DateTime dt = DateTime.Now;
string st = String.Format("{0:yyyyMMdd}", dt);//20161120
Task[] myTasks = new Task[list.Count];
int i = 0;
foreach (string item in list)
{
     if (item.StartsWith("GExport_") && (!item.ToUpper().Contains("DUM")) && (item.Contains(st)) && (!item.ToUpper().Contains("BLK")))
     {
         4gpath = item;
        //Downloadfile()   
        ftp.Get(dtr["REMOTE_FILE_PATH"].ToString() + 4gpath , @localDestnDir + "\\" + dtr["SOURCE_PATH"].ToString());
        download_location_hw = dtr["LOCAL_FILE_PATH"].ToString();
        // Spin off a background task to process the file we just downloaded
        myTasks[i++] = Task.Factory.StartNew(() =>
        {
            //Extractfile()             
            ExtractZipfiles(download_location_hw + "//" + huwawei4gpath, dtr["REMOTE_FILE_PATH"].ToString(), 
             dtr["FTP_SERVER"].ToString(), dtr["FTP_USER_ID"].ToString(),
             dtr["TECH_CODE"].ToString(), dtr["VENDOR_CODE"].ToString());
        }
    }
}

private static void ExtractZipfiles(string download_location_hw,string remotepath,string server,string userid,string tech,string vendor)
{
    if (download_location_hw.Contains(".gz"))
    {
        DirectoryInfo directorySelected = new DirectoryInfo(Path.GetDirectoryName(download_location_hw));
        foreach (FileInfo fileToDecompress in directorySelected.GetFiles("*.gz"))
        {
            Decompress(fileToDecompress);
        }
    }

}

2 个答案:

答案 0 :(得分:2)

您可能会在相同的文件夹中提取已下载的zip个文件:

download_location_hw = dtr["LOCAL_FILE_PATH"].ToString();
var directorySelected = new DirectoryInfo(Path.GetDirectoryName(download_location_hw));

如果出于某种原因,您将两个文件定位到同一本地路径,则最终foreach语句将与其他任务交叉:

foreach (var fileToDecompress in directorySelected.GetFiles("*.gz"))
{
    Decompress(fileToDecompress);
}

在这里,您要搜索文件夹中的所有*.gz个文件,即使是那些仍然处于下载模式或正在被其他进程解压缩的文件。因此,您应该确定每个任务都在不同的文件夹中运行。

如果你想简单修改你的代码,你可以试试这个(这里的主要改变不是枚举所有的档案文件,而只是从任务参数中得到一个,FileInfo constructor):

private static void ExtractZipfiles(string download_location_hw,string remotepath,string server,string userid,string tech,string vendor)
{
    if (download_location_hw.Contains(".gz") && File.Exists(download_location_hw))
    {
        Decompress(new FileInfo(download_location_hw));
    }
}

答案 1 :(得分:1)

各种任务在同一个下载路径上运行并导致冲突。 例如。任务1放置a.gz,b.gz,而任务2放置c.gz在" MyDir"。

但现在两个任务都在运行,并尝试在解压缩过程中将文件重命名为 a,b,c 。因此,您似乎需要为每个任务创建子目录,或者使用唯一标识符前缀\ suffix下载的文件名。