【问题标题】:Background worker, and cross thread issue后台工作者和跨线程问题
【发布时间】:2011-06-12 03:47:31
【问题描述】:

我有一个运行良好的 Winforms 应用程序。在处理设备的串行数据期间,使用 BackgroundWorkerThread 来管理 GUI 可用性。

一切正常。

现在,我正在添加一个新方法,并复制我在其他形式中所做的。但是我遇到了跨线程异常。

我这样声明我的 BWT:

BackgroundWorker bw = new BackgroundWorker();
            bw.WorkerSupportsCancellation = true;
            bw.DoWork += DownloadGpsDataFromDevice;
            bw.WorkerReportsProgress = true;
            bw.RunWorkerAsync();

然后我有一个像这样声明的方法,它在后台工作:

private void DownloadGpsDataFromDevice(object sender, DoWorkEventArgs e)
{
    _performScreenUpdate = true;
    tsStatus.Text = "Downloading GPS Data...";
    Invalidate();
    Refresh();


    Common.WriteLog("Extracting raw GPS data. Sending LL.");

    ReplyString raw = DeviceServices.ExecuteCommand("$LL");

DeviceServices.ExecuteCommand("$LL");是完成工作的位,但我在上一行遇到异常,我在其中登录到文本文件。现在,这让您担心 - 写入文件。但是,我在另一个 BWT 中已经这样做了数千次。

我使写作线程安全。这是我的 Common.WriteLog 方法:

public static void WriteLog(string input)
{
    lock (_lockObject)
    {
        WriteLogThreadSafe(input);
    }
}

private static void WriteLogThreadSafe(string input)
{
    Directory.CreateDirectory(LogFilePath);
    StreamWriter w = File.AppendText(LogFilePath + @"\" + LogFileName);
    try
    {
        w.WriteLine(string.Format("{0}\t{1}", DateTime.Now, input));
    }
    catch (Exception e)
    {
        System.Console.WriteLine("Error writing to log file!");
        System.Console.WriteLine("Tried to write: [" + input + "]");
        System.Console.WriteLine("Failed with error: [" + e.Message + "]");
    }
    finally
    {
        w.Close();
    }

}

这已经工作了很长时间。我不相信错误存在。我想我可能只是在通话中遗漏了什么?

【问题讨论】:

    标签: c# multithreading backgroundworker


    【解决方案1】:

    您不能从 BackgroundWorker 线程更改 UI 元素。您必须通过调用 Invoke() 来编组回 UI 线程。

    试试这个

    private void DownloadGpsDataFromDevice(object sender, DoWorkEventArgs e)
    {
        _performScreenUpdate = true;
        Invoke((MethodInvoker)(() => {
                 tsStatus.Text = "Downloading GPS Data...";
                 Invalidate();
                 Refresh();
         });
         ...
    

    【讨论】:

    • 这些行似乎可以正常工作...还是我收到错误错误?它在 Common.WriteLog("Extracting raw GPS data. Sending LL."); 上崩溃
    • 等等……你是对的,看来!该应用程序在 LogLine 上崩溃了,但是当我注释掉您提到的那些行时……它可以工作。奇怪的是它在日志行上崩溃了。谢谢!
    • @cdotlister WriteLog() 对我来说似乎没问题,不应该导致跨线程异常。在那种情况下我不确定。
    【解决方案2】:

    问题是您正在从非 UI 线程更新 UI 元素:

    这些行不应在DownloadGpsDataFromDevice

    tsStatus.Text = "Downloading GPS Data...";
    Invalidate();
    Refresh();
    

    利用BackgroundWorker 运行方法bw.ReportProgress(0);。更新专门为此目的设计的 ProgressChanged 处理程序中的 UI。

    void bw_ProgressChanged(object sender, ProgressChangedEventArgs e)
    {
        if (e.ProgressPercentage = 0)
        {
            tsStatus.Text = "Downloading GPS Data...";
            Invalidate();
            Refresh();
        }
    }
    

    【讨论】:

    • 这些行似乎可以正常工作...还是我收到错误错误?它在 Common.WriteLog("Extracting raw GPS data. Sending LL."); 上崩溃
    • 等等……你是对的,看来!该应用程序在 LogLine 上崩溃了,但是当我注释掉你提到的那些行时......它可以工作。奇怪的是它在日志行上崩溃了。谢谢!
    • @cdotlister - 如果您不使用 ProgressChangedRunWorkerCompleted 事件,则无需使用。这两个事件是BackgroundWorker 的重点。这些事件总是在 UI 线程中触发。
    【解决方案3】:

    某些实例不能或不应被多个线程访问。您有两个选项可以保护您的数据免受跨线程异常的影响。

    您可以在使用锁从多个线程访问对象时锁定您的对象:

    对象储物柜 = 新对象();

    SomeObject MyObject = new SomeObject();

    private void FromMultipleThread()
    {
        lock(locker)
        {
            MyObject = OtherObject;
        }
    }
    

    您的第二个选项是使用 ManualResetEvent 锁定您的线程。这很简单,您只需从 ManualResetEvent 调用 WaitOne() 即可锁定您的线程,而其他线程访问您的“跨线程”对象。

    在您的情况下,您可能希望从 backgroundWorker 的 reportProgress 更改您的 UI。 reportProgress 将回到初始线程,然后您可以修改您的 UI。

    【讨论】:

      【解决方案4】:

      您确定这是正确的路线吗?我认为您不应该能够更新工作人员中的 ui。尝试注释掉 gui 更新并清理并构建您的解决方案,以查看日志记录是否真的是问题所在。要更新 ui,请设置 WorkerReportsProgress 并为其创建一个事件处理程序以更新 ui 并报告 worker 中的进度。

      【讨论】:

        猜你喜欢
        • 1970-01-01
        • 1970-01-01
        • 2011-06-18
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        相关资源
        最近更新 更多