问题 简单的BackgroundWorker不会更新网页上的标签


我使用了一段有用的简单代码 岗位

它使用一个按钮和一个标签,标签应报告“10%完成”......“20%完成”......等等。

当我调试时,代码被点击,但我的标签没有在浏览器上更新。

我尝试使用和不使用更新面板,有和没有母版页。

protected void btnStartThread_Click(object sender, EventArgs e)
    {
        BackgroundWorker bw = new BackgroundWorker();

        // this allows our worker to report progress during work
        bw.WorkerReportsProgress = true;

        // what to do in the background thread
        bw.DoWork += new DoWorkEventHandler(delegate(object o, DoWorkEventArgs args)
        {
            BackgroundWorker b = o as BackgroundWorker;

            // do some simple processing for 10 seconds
            for (int i = 1; i <= 10; i++)
            {
                // report the progress in percent
                b.ReportProgress(i * 10);
                Thread.Sleep(1000);
            }
        });



        // what to do when progress changed (update the progress bar for example)
        bw.ProgressChanged += new ProgressChangedEventHandler(delegate(object o, ProgressChangedEventArgs args)
        {
            label1.Text = string.Format("{0}% Completed", args.ProgressPercentage);
        });


        // what to do when worker completes its task (notify the user)
        bw.RunWorkerCompleted += new RunWorkerCompletedEventHandler(
        delegate(object o, RunWorkerCompletedEventArgs args)
        {
            label1.Text = "Finished!";
        });

        bw.RunWorkerAsync();

4022
2018-04-29 12:31


起源

如果HTML已经通过网络发送到客户端,您希望如何更新它?您需要在客户端上使用AJAX来更新页面。 - Martin Costello
这就是为什么我认为页面上的脚本管理器和更新面板可以解决这个问题:/ - wotney
这应该。我不记得了,你确定你不需要调用像updatePanel.Update();或者类似的? - Falgantil
您需要使用SignalR之类的信息将该信号发送给客户端。 - Mike Perrenoud
在客户端服务器Web appalication中,服务器不会启动与客户端的通信,它只响应客户端的请求。通过使用ajax请求或使用signalr的服务器推送模式,您可以通过客户端拉动模式实现所需 asp.net/signalr/overview/guide-to-the-api/... - lyz


答案:


您可以进行长时间轮询以获取更新的结果。我个人不会在Web应用程序中使用后台工作程序。另外,请考虑查看SignalR,服务器发送事件和异步和等待或任务并行库。


5
2018-04-29 12:37



有点帮助,但不是很多答案。我真的需要弄清楚如何在长时间运行的任务中将更新返回到网页,返回给用户。 - AgapwIesu


BackgroundWorker 不适合ASP.NET,并且不能很好地与ASP.NET一起使用,因为ASP.NET在请求/响应模型上工作。要接收任何更新,客户端的浏览器需要从服务器请求信息。比如你的 Label 要更新,您需要向服务器发送其新值的请求。

一个选项可能是在客户端上使用AJAX来更新页面。但是,不是使用正式的AJAX,而是可以使用 iframe 这是一种异步执行长时间运行进程的好方法,而不依赖于任何特定的AJAX框架。单击按钮时,看不见 iframe 动态生成并用于执行 LongRunningProcess.aspx 在后台:

<head runat="server">
<script type="text/javascript">
    function BeginProcess() {
        var iframe = document.createElement("iframe");
        iframe.src = "LongRunningProcess.aspx";
        iframe.style.display = "none";
        document.body.appendChild(iframe);
        document.getElementById('trigger').disabled = true;
    }

    function UpdateProgress(PercentComplete, Message) {
        document.getElementById('<%= Label1.ClientID %>').innerHTML = PercentComplete + '% ' + Message;
}
</script>
</head>

<body>
   <form id="form1" runat="server">
   <div>
    <input type="submit" value="Start Long Running Process" id="trigger" 
       onclick="BeginProcess(); return false;" style="width: 185px;"/>
    <asp:Label ID="Label1" runat="server" Text="Label"></asp:Label>
   </div>
   </form>
</body>

LongRunningProcess.aspx.cs

protected void Page_Load(object sender, EventArgs e)
{
   for (int i = 1; i <= 9; i++)
   {
       UpdateProgress(i * 10, " Completed.");
       System.Threading.Thread.Sleep(1000);
   }
   UpdateProgress(100, "Finished!");
}

protected void UpdateProgress(int PercentComplete, string Message)
{
    Response.Write(String.Format("<script type=\"text/javascript\">parent.UpdateProgress({0}, '{1}');</script>", PercentComplete, Message));
    Response.Flush();
}

您可以在Dave Ward的精彩文章中看到结果并阅读更多内容:   长期请求的轻松增量状态更新。


4
2017-10-28 16:36





这种情况很少见,但.NET有时可以同步运行异步任务(从MSDN上的文档中可能会发生这种情况并不清楚)但如果发生这种情况,它将阻止WinForms应用程序中的消息循环。

要测试这个理论,您可以尝试在ProgressChanged处理程序和DoWork处理程序中调用Application.DoEvents(),看看是否有帮助。这是一个丑陋的工作,但它应该强制UI更新如果disptacher同步运行您的任务。


3
2017-10-30 13:26



这显然是Web窗体,而不是Windows窗体。 - areyesram


你基本上有两个选择。

一,您可以使用JavaScript从计时器上的浏览器轮询以获得当前完成百分比。 setInterval 是一种显而易见的使用方法,您可以使用jQuery或类似方法对服务器进行AJAX调用。但这有点效率低,因为你的服务器听起来好像知道它何时进行或完成,所以服务器应该引发事件。

两个,你想要的更多,是使用SignalR。当提出定时事件时,你已经设置了从服务器“推送”的东西,但是在HTTP中没有简单的方法(HTML5 WebSockets也是另一种选择)。使用像SignalR这样的服务器推送技术,您可以创建向客户端引发事件的“集线器”。它们可以是特定于用户的目标事件或所有客户类型的事件 - 由您决定。

在这里查看介绍/演示。他们很容易解决一个相当难的问题: http://www.asp.net/signalr

另外,正如其他人所提到的,您可能不想使用BackgroundWorker。尝试QueueBackgroundWorkItem。 http://blogs.msdn.com/b/webdev/archive/2014/06/04/queuebackgroundworkitem-to-reliably-schedule-and-run-long-background-process-in-asp-net.aspx


2
2017-10-30 18:04