【问题标题】:stop a thread before closing form在关闭表单之前停止线程
【发布时间】:2016-10-24 19:43:33
【问题描述】:

使用 .NET4、c#、 我在winform上有一个网格。当我点击一个进程按钮时,我需要对每一行执行几个步骤的及时操作,然后用结果更新该行。这样,用户可以在处理下一行时看到进度。如果用户点击取消或关闭表单,我需要完成当前行的过程,然后中断或关闭表单。根据我的研究,这是我想出的。我还有 2 个问题。

这是我的代码:

delegate void OperateGridMethodDelegate();
delegate void UpdateProgressDelegate(int rowIndex, string msg);
bool isCancelled = false;


private void ButtonOne_Click(object sender, EventArgs e)
{
    isCancelled=false;
    OperateGridMethodDelegate delegate = new OperateGridMethodDelegate(LongProcess);`
    delegate.BeginInvoke(null,null);

}

private void ButtonTow_Click(object sender, EventArgs e)
{

    MyButtonOne.Enabled=false;
    isCancelled=true;

}

private void LongProcess()
{

    foreach (DataRow row in dataTableOne.Rows)
    {
        //Do Lengthy Process

        DisplayResult( rowIndex, "this is result of operation" );
        if(isCancelled)
        {
            MyButtonOne.Enabled=true;
            break;

        }

    }

}

private void DisplayResult(int rowIndex,string message)
{

    if(myGrid.InvokeRequired == false)
    {

        myGrid.Rows[rowIndex].value = message;

    }

    else
    {

        UpdateProgressDelegate delegate = new  UpdateProgressDelegate(DisplayResult);
        myGrid.BeginInvoke(delegate, new object[] { rowIndex, message });

    }

}

我剩下的两个问题是:

  1. 当表单关闭时,我想等待 LongProcess(由我的 OperateGridMethodDelegate 调用)完成它的行,然后取消(与 ButtonTwo 一样),但我可以'让这个工作。

  2. 我在调试时注意到一个奇怪的行为,我不确定这是一个错误还是正确的,当一行完成处理时,结果没有设置。相反,处理下一行,然后显示上一行结果。当第三行结束时显示第二行结果等。然后最后两行结果基本同时显示。

我的方法正确吗?

【问题讨论】:

  • 不要使用委托的 BeginInvoke() 方法来实现线程,bool 不是同步原语。有更好的方法可以做到这一点,this Q+A 中介绍了如何正确处理 BackgroundWorker。

标签: c# multithreading asynchronous invoke begininvoke


【解决方案1】:

如果你在一个线程中启动了一些任务,等待它完成后再继续你可以这样做:

yourThread.Join();

这将阻塞直到线程退出。您可能想在 Form_Closing 事件中使用类似的东西

或者,如果您希望线程在应用程序终止时自动终止,您需要设置线程的背景属性:

yourThread.IsBackground = true;

这意味着它是您的主 (UI) 线程的后台线程,因此它将以主线程结束。

编辑:

使用你自己的线程:

private Thread yourThread = new Thread(() = > LongProcess());
yourThread.IsBackground = true;
yourThread.Start();  // This will begin your 'LongProcess' function.

现在如果你想在某个地方阻塞并等待它完成,你可以做我首先提到的:yourThread.Join();

另请注意,如果您设置了IsBackground = false 并且有人关闭了应用程序,则该线程不会立即退出,它会一直持续到完成为止,尽管 UI 窗口已关闭。

【讨论】:

  • 你的线程是什么?我在哪里得到这个?我没有完全遵循
  • 我在答案中添加了更多内容。如果你想控制线程,你应该使用你自己的Thread对象
  • 你是不是想说,而不是我的代码 - private void ButtonOne_Click(object sender, EventArgs e) { isCancelled=false; OperateGridMethodDelegate 委托 = 新的 OperateGridMethodDelegate(LongProcess);`delegate.BeginInvoke(null,null); },我应该这样做吗?私人无效ButtonOne_Click(对象发送者,EventArgs e){isCancelled=false;私有线程 yourThread = new Thread(() = > LongProcess());你的线程.IsBackground = true;你的线程.Start(); }
  • 我不会按照您通常的方式进行操作,但考虑到您想要对长期运行的任务进行控制,使用您自己的Thread 绝对是可取的。您可能会在声明 delegate void 的位置声明线程,并在 ButtonOne_Click 事件中声明 .Start() 线程
  • 你介意告诉我你会如何建议这样做吗?我不打算只这样做?
猜你喜欢
  • 2013-01-31
  • 2020-05-29
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2012-10-23
  • 1970-01-01
  • 2011-04-30
  • 1970-01-01
相关资源
最近更新 更多