在操作完成之前,UI不响应

问题描述

我不确定标题是否很好地说明了此问题。本质上,我所拥有的是一个WinForm应用程序,该应用程序将文件夹中的文件列表检索到ListView中,然后单击按钮以通过FTP将其上传到远程服务器。

功能上讲,该应用程序可以按预期运行:

  1. 打开应用
  2. 在ListView控件中查看文件列表
  3. 点击“上传”按钮
  4. ListView中列出的文件上传;每次成功上传后,ListView都会更新为显示 “成功”
  5. 所有文件上传后,操作停止。

我的问题是,单击上传按钮后,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

选择更适合您的选项。我相信对于没有线程编程经验的人来说,我的方法更容易掌握。