问题描述
我不确定标题是否很好地说明了此问题。本质上,我所拥有的是一个WinForm应用程序,该应用程序将文件夹中的文件列表检索到ListView中,然后单击按钮以通过FTP将其上传到远程服务器。
从功能上讲,该应用程序可以按预期运行:
我的问题是,单击上传按钮后,UI几乎没有响应,直到操作完成。每个文件上传后,ListView都会按预期更新,甚至可以使活动行保持焦点。这是处理文件的for循环。有一点背景-在下面的代码摘录中,每个for ... loop处理2个文件-主文件是ListView中显示的唯一文件。每个循环中的第二个文件是一个触发器文件,该文件在发送其主文件后即发送。即:.primary,.trigger。两个文件都必须发送才能注册成功。如果主文件没有相应的触发文件,则ListView中将无法使用该文件进行上传。
foreach (ListViewItem item in lvSourceFiles.Items)
{
int rowIndex = item.Index;
string fileName = item.SubItems[2].Text;
lvSourceFiles.EnsureVisible(rowIndex);
transferStatus = "Failed"; // Set this as a default
// Transfer the source file first
transferResult = session.PutFiles(readyFile,destFile,false,transferOptions);
// Throw on any error
transferResult.Check();
// If the source file transfer was successful,then transfer the trigger file
if (transferResult.IsSuccess)
{
transferResult = session.PutFiles(triggerFile,transferOptions);
transferResult.Check();
if (transferResult.IsSuccess)
{
transferStatus = "Success";
}
}
UpdateResultsToListView(lvSourceFiles,rowIndex,fileName,transferStatus);
}
是这种情况下,我需要实现某种异步功能,还是有更好的方法来做到这一点,以使UI在上载过程中不会冻结?本质上,我希望能够在上传运行时与表单进行交互,例如具有取消按钮来停止上传。就目前而言,在作业完成或终止应用程序之前,我无法对表单进行任何操作。
谢谢, 詹姆斯
解决方法
您可以使用async / await和方便的Task.Run
方法将长时间运行的操作卸载到ThreadPool
线程中:
transferResult = await Task.Run(() => session.PutFiles(readyFile,destFile,false,transferOptions));
...并且:
transferResult = await Task.Run(() => session.PutFiles(triggerFile,transferOptions));
您还应该在事件处理程序中添加async
修饰符,以启用await
运算符。
重要:避免在卸载的方法中执行任何与UI相关的操作。如果要在操作过程中与UI进行通信,例如progress reporting,请使用Progress<T>
类。
您不能在GUI线程上进行冗长的操作。在后台线程上执行它们。
@Theodor的回答正确表明您可以将PutFiles
移动到线程池中。
另一种选择是将所有上传逻辑移至线程池,并使用Control.Invoke
回调主线程,仅用于GUI更新。
有关完整示例,请参见WinSCP文章Displaying FTP/SFTP transfer progress on WinForms ProgressBar。
选择更适合您的选项。我相信对于没有线程编程经验的人来说,我的方法更容易掌握。