Task.WaitAll(...)阻止MVC 4应用程序

时间:2013-07-23 09:18:34

标签: asp.net-mvc-4 io task-parallel-library async-await

在MVC 4应用程序中,我有一个将文件复制到目标文件夹的任务。因为我有多个文件,所以我为每个文件创建一个任务,我想等到它们都完成了。问题是我Task.WaitAll(copyingTasks.ToArray())处的代码块好像任务永远不会结束,所以它永远不会通过那行代码。下面是示例代码:

    private void CopyFilesFromWorkingCopyForProject(string projectName)
    {
        var copyingTasks = new List<Task>
            {
                CopyAllFromDirectoryToDirectory(FilesUtils.AndroidConfigsPath(), FilesUtils.AndroidPathForProject(projectName)),
                CopyAllFromDirectoryToDirectory(FilesUtils.AndroidValuesPath(), FilesUtils.AndroidPathForProject(projectName)),
                CopyFileToDirectory(FilesUtils.AndroidManifestPath(), FilesUtils.AndroidPathForProject(projectName)),
                CopyAllFromDirectoryToDirectory(FilesUtils.IosConfigsPath(), FilesUtils.IosPathForProject(projectName))

            };
        Task.WaitAll(copyingTasks.ToArray());
    }

    private async Task CopyAllFromDirectoryToDirectory(string sourceDirectory, string destinationDirectory)
    {
        foreach (string filename in Directory.EnumerateFiles(sourceDirectory))
        {
            await CopyFileToDirectory(filename, destinationDirectory);
        }
    }

    private async Task CopyFileToDirectory(string filename, string destinationDirectory)
    {
        using (FileStream sourceStream = File.Open(filename, FileMode.Open))
        {
            using (FileStream destinationStream = File.Create(destinationDirectory + filename.Substring(filename.LastIndexOf('\\'))))
            {
                await sourceStream.CopyToAsync(destinationStream);
            }
        }
    }

如果我评论Task.WaitAll(copyingTasks.ToArray());它不再阻止,但我想等待所有文件被复制。

1 个答案:

答案 0 :(得分:11)

Combining await and synchronous wait leads to deadlocks,因为async方法会尝试在当前被您等待阻止的上下文中恢复。

你应该做的是让CopyFilesFromWorkingCopyForProject()async(以及调用它的方法,以及调用它的方法,......):

private async Task CopyFilesFromWorkingCopyForProject(string projectName)
{
    var copyingTasks = new List<Task>
        {
            CopyAllFromDirectoryToDirectory(FilesUtils.AndroidConfigsPath(), FilesUtils.AndroidPathForProject(projectName)),
            CopyAllFromDirectoryToDirectory(FilesUtils.AndroidValuesPath(), FilesUtils.AndroidPathForProject(projectName)),
            CopyFileToDirectory(FilesUtils.AndroidManifestPath(), FilesUtils.AndroidPathForProject(projectName)),
            CopyAllFromDirectoryToDirectory(FilesUtils.IosConfigsPath(), FilesUtils.IosPathForProject(projectName))

        };
    await Task.WhenAll(copyingTasks);
}

如果您不能或不想这样做,则需要确保async方法不会在当前上下文中恢复。为此,您可以对所有ConfigureAwait(false)使用await,也可以使用async在后​​台线程上调用Task.Run()方法。