Улавливание исключений из другого потока

У меня есть метод, выполняющийся в отдельном потоке. Поток создается и запускается из формы в приложении Windows. Если исключение выбрасывается из потока, то как лучше всего передать его в основное приложение. Прямо сейчас, я передаю ссылку на основную форму в поток, затем вызывая метод из потока и заставляя метод вызываться по основному потоку приложения. Есть ли лучший способ сделать это, потому что мне не нравится, как я это делаю сейчас.

Пример моей формы:

public class frmMyForm : System.Windows.Forms.Form
{
    /// <summary>
    /// Create a thread
    /// </summary>
    /// <param name="sender"></param>
    /// <param name="e"></param>
    private void btnTest_Click(object sender, EventArgs e)
    {
        try
        {
            //Create and start the thread
           ThreadExample pThreadExample = new ThreadExample(this);
           pThreadExample.Start();
        }
        catch (Exception ex)
        {
            MessageBox.Show(ex.Message, Application.ProductName);
        }
    }

    /// <summary>
    /// Called from inside the thread 
    /// </summary>
    /// <param name="ex"></param>
    public void HandleError(Exception ex)
    {
        //Invoke a method in the GUI main thread
        this.Invoke(new ThreadExample.delThreadSafeTriggerScript(HandleError), new Object[] { ex });
    }

    private void __HandleError(Exception ex)
    {
        MessageBox.Show(ex.Message);
    }
}

Пример моего класса потока:

public class ThreadExample
{
    public delegate void delThreadSafeHandleException(System.Exception ex);

    private Thread thExample_m;

    frmMyForm pForm_m;
    private frmMyForm Form
    {
        get
        {
            return pForm_m;
        }
    }

    public ThreadExample(frmMyForm pForm)
    {
        pForm_m = pForm;

        thExample_m = new Thread(new ThreadStart(Main));
        thExample_m.Name = "Example Thread";
    }

    public void Start()
    {
        thExample_m.Start();
    }

    private void Main()
    {
        try
        {
            throw new Exception("Test");
        }
        catch (Exception ex)
        {
            Form.HandleException(ex);
        }
    }
}

Ответы

Ответ 1

Итак, вы используете Invoke для перехода к потоку пользовательского интерфейса, по внешнему виду - именно это вам и нужно делать. Я лично использовал Action < Exception для простоты и, возможно, BeginInvoke вместо Invoke, но в основном вы поступаете правильно.

Ответ 2

Вместо этого используйте класс BackgroundWorker в платформе .NET. Это лучшая практика для выполнения пользовательского интерфейса в другом потоке.

Ответ 3

Вероятно, лучшим способом было бы передать делегат в поток вместо ссылки на его форму.

Ответ 4

Бросание исключений между потоками непросто и, вероятно, нежелательно. вместо этого вы можете передать исключение, используя общую структуру данных или переменную, и использовать waitHandle для ожидания в первом потоке.

Ответ 5

Я полностью согласен с Дрором. Формально мы можем назвать эту структуру как FaultContract. В принципе, когда в другом потоке произошло исключение, поток клиентов вряд ли может сделать что-то в тот момент, за исключением того, что собирать эту информацию и действовать соответствующим образом в ней. Если они есть в разных AppPool, тогда существует дополнительная сложность Serialization (это может быть отдельная тема в целом).