Directory.Move():拒绝访问路径

时间:2013-09-20 19:53:34

标签: c# exception-handling directory access-denied

我正在使用C#在Visual Studio 2010中编写此Windows窗体应用程序。

表单上有一个Execute按钮,用户将点击该按钮,程序将生成一些文件并存储在Output文件夹中(由程序使用Directory.CreateDirectory()创建)

我想创建一个Archive文件夹来保存以前运行的输出文件。

在每次运行开始时,我尝试将现有的Output文件夹移动到Archive文件夹,然后创建一个新的Output文件夹。下面是我跑去移动目录的功能。

static void moveToArchive()
{
    if (!Directory.Exists("Archive")) Directory.CreateDirectory("Archive");
    string timestamp = DateTime.Now.ToString("yyyyMMddHHmms");
    try
    {
        Directory.Move("Output", "Archive\\" + timestamp);
    }
    catch(Exception e)
    {
        Console.WriteLine("Can not move folder: " + e.Message);
    }
}

我遇到的问题让我很困惑......

有些时候我可以成功地将Output文件夹移动到存档,但有时会失败。

我从捕获异常中得到的错误消息是Access to path 'Output' is denied.

我检查过Output文件夹中的所有文件都没有被使用。我不明白访问权限有时被拒绝,而不是所有时间都被拒绝。

有人可以向我解释并告诉我如何解决问题吗?

- 编辑 -

在HansPassant评论之后,我稍微修改了一下这个函数以获取当前目录并使用完整路径。但是,我仍然遇到同样的问题。

该功能现在如下所示:

static void moveToArchive()
{
    string currentDir = Environment.CurrentDirectory;
    Console.WriteLine("Current Directory = " + currentDir);
    if (!Directory.Exists(currentDir + "\\Archive")) Directory.CreateDirectory(currentDir + "\\Archive");
    string timestamp = DateTime.Now.ToString("yyyyMMddHHmms");
    try
    {
        Directory.Move(currentDir + "\\Output", currentDir + "\\Archive\\" + timestamp);
    }
    catch(Exception e)
    {
        Console.WriteLine("Can not move folder: " + e.Message);
    }
}

我打印出当前目录,它正如我所期待的那样,我仍然无法使用完整路径。 Access to path 'C:\Users\Me\Desktop\FormApp\Output' is denied.

- 编辑 -

感谢大家的回答和评论。

我想你们有些人会错过这一部分所以我会更加强调它。

Directory.Move()有时会工作,有时会失败。

当功能成功时,没有问题。 Output文件夹已移至Archive

当函数失败时,我得到的异常消息是访问路径被拒绝。

6 个答案:

答案 0 :(得分:9)

谢谢大家的回复和帮助。我弄清楚了问题是什么。

这是因为有一个文件没有完全关闭。

我正在检查生成的文件,并错过了程序正在读取的文件。

生成的所有文件都已完全关闭。这是我使用StreamReader打开但未关闭的一个文件。我修改了代码,现在没有问题,所以我认为这是问题所在。

感谢所有的评论和答案,这无疑帮助我思考并解决问题。

答案 1 :(得分:2)

请参阅http://windowsxp.mvps.org/processlock.htm

有时,您尝试移动或删除文件或文件夹,并接收访问冲突或正在使用的文件 - 错误。要成功删除文件,您需要确定锁定文件的进程。您需要先退出该进程,然后删除该特定文件。要知道哪个进程已锁定文件,您可以使用本文中讨论的方法之一。

使用Process Explorer - 从http://download.sysinternals.com/files/ProcessExplorer.zip

下载

Process Explorer显示有关已打开或加载哪些句柄和DLL进程的信息。

从Microsoft站点下载Process Explorer并运行该程序。 单击“查找”菜单,然后选择“查找句柄”或“DLL”... 键入文件名(由某个进程锁定的文件的名称。) 键入搜索短语后,单击“搜索”按钮 您应该看到正在访问该文件的应用程序列表。

答案 2 :(得分:1)

我最近碰到了同样的问题。使用PE我认为只有使用该特定目录的进程才是explorer.exe。我用浏览器打开了几个窗口,一个指向我即将移动的一个父目录。

看来,在我访问该子文件夹然后返回(甚至到根级别!)后,句柄仍然被资源管理器保留,因此C#无法以任何方式修改它(更改标志,属性等) )。

为了让C#正常运行,我不得不杀死那个资源管理器窗口。

答案 3 :(得分:0)

试试这个: 如果这没有解决,可能会检查/更改防病毒软件,或者某些其他程序正在锁定某个文件或文件夹。

    static object moveLocker = new object();
    static void moveToArchive()
    {
        lock (moveLocker)
        {
            System.Threading.Thread.Sleep(2000);  // Give sometime to ensure all file are closed.

            //Environment.CurrentDirectory = System.AppDomain.CurrentDomain.BaseDirectory;
            string applicationPath = System.AppDomain.CurrentDomain.BaseDirectory;
            string archiveBaseDirectoryPath = System.IO.Path.Combine(applicationPath, "Archive");

            if (!Directory.Exists(archiveBaseDirectoryPath)) Directory.CreateDirectory(archiveBaseDirectoryPath);

            String timestamp = DateTime.Now.ToString("yyyyMMddHHmms");
            String outputDirectory = System.IO.Path.Combine(Environment.CurrentDirectory, "Output");
            String destinationTS = System.IO.Path.Combine(archiveBaseDirectoryPath, timestamp);
            try
            {
                Directory.Move(outputDirectory, destinationTS);                
            }
            catch (Exception ex)
            {
                Console.WriteLine("Can not move folder " + outputDirectory + " to: " + destinationTS + "\n" + ex.Message);
            }
        }
    }

答案 4 :(得分:0)

File.SetAttributes(Application.dataPath + "/script", FileAttributes.Normal);
Directory.Move(Application.dataPath + "/script", Application.dataPath + "/../script");

这解决了我的问题。

答案 5 :(得分:0)

我遇到了同样的问题,有时会失败,但并非总是如此。我以为可以将其包装在Try Catch块中,并向用户显示“访问被拒绝”消息,一旦将其包装在Try Catch块中,它就不会失败。我无法解释原因。

 If existingFile.FileName <> newFileName Then
    Dim dir As New IO.DirectoryInfo(existingFile.FilePath)
    Dim path As String = System.IO.Path.GetDirectoryName(dir.FullName)
    newFileName = path & "\" & newFileName

    File.SetAttributes(existingFile.FilePath, FileAttributes.Normal)
        Try
            IO.File.Move(existingFile.FilePath, newFileName)
        Catch ex As Exception

        End Try
End If