11

我正在尝试使用 SSH.NET 从 SFTP 服务器异步下载文件。如果我同步执行,它可以正常工作,但是当我异步执行时,我会得到空文件。这是我的代码:

var port = 22;
string host = "localhost";
string username = "user";
string password = "password";
string localPath = @"C:\temp";

using (var client = new SftpClient(host, port, username, password))
{
    client.Connect();
    var files = client.ListDirectory("");

    var tasks = new List<Task>();

    foreach (var file in files)
    {                        
        using (var saveFile = File.OpenWrite(localPath + "\\" + file.Name))
        {
            //sftp.DownloadFile(file.FullName,saveFile); <-- This works fine
            tasks.Add(Task.Factory.FromAsync(client.BeginDownloadFile(file.FullName, saveFile), client.EndDownloadFile));
        }                        
    }

    await Task.WhenAll(tasks);
    client.Disconnect();

}
4

1 回答 1

13

因为saveFile是在using块中声明的,所以在你启动任务后它就被关闭了,所以无法完成下载。实际上,我很惊讶您没有例外。

您可以提取代码以下载到单独的方法,如下所示:

var port = 22;
string host = "localhost";
string username = "user";
string password = "password";
string localPath = @"C:\temp";

using (var client = new SftpClient(host, port, username, password))
{
    client.Connect();
    var files = client.ListDirectory("");

    var tasks = new List<Task>();

    foreach (var file in files)
    {                        
        tasks.Add(DownloadFileAsync(file.FullName, localPath + "\\" + file.Name));
    }

    await Task.WhenAll(tasks);
    client.Disconnect();

}

...

async Task DownloadFileAsync(string source, string destination)
{
    using (var saveFile = File.OpenWrite(destination))
    {
        var task = Task.Factory.FromAsync(client.BeginDownloadFile(source, saveFile), client.EndDownloadFile);
        await task;
    }
}

这样,在您完成下载文件之前不会关闭文件。


查看 SSH.NET 源代码,看起来异步版本DownloadFile没有使用“真正的”异步 IO(使用 IO 完成端口),而是在新线程中执行下载。BeginDownloadFile所以使用/没有真正的优势EndDownloadFile;您不妨DownloadFile在自己创建的线程中使用:

Task DownloadFileAsync(string source, string destination)
{
    return Task.Run(() =>
    {
        using (var saveFile = File.OpenWrite(destination))
        {
            client.DownloadFile(source, saveFile);
        }
    }
}
于 2015-12-02T10:05:52.980 回答