Улавливание исключений из другого потока
У меня есть метод, выполняющийся в отдельном потоке. Поток создается и запускается из формы в приложении 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 (это может быть отдельная тема в целом).