はじめに
タイトルの通りです。
今時 .NET Framework の Windows.Forms で新規開発案件とかあるんですか?と聞かれそうですが、4年ほど受託開発してきて担当案件の9割9分はこれでしたし、半年以上先の案件/開発予定もこれだけで埋まっている状況です...!
この感じがまだしばらく続くのかなと思ったので、コピペ用に書き残しておきます。
2024-04-20 追記
たくさんのコメントありがとうございました...!
頂いた内容の方法でも試そうと、ひとつずつ機能の差異なども調べていたのですが、自分にはまだ理解が追いつかない部分もあり(くやしいくやしい...)、少し沼ってきたため、追記という形ではなくどこかで個別に新規記事として投稿できればと思います...
環境
- Microsoft .NET Framework 4.8.1
- Windows.Forms
ソース
例外捕捉時、メッセージを表示していますが、実際はログ出力にすることが多いです。
using System;
using System.Text;
using System.Threading;
using System.Windows.Forms;
namespace App
{
internal static class Program
{
[STAThread]
static void Main()
{
Application.ThreadException += Application_ThreadException;
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
var mutex = new Mutex(true, Application.ProductName, out bool createdNew);
try
{
if (!createdNew)
{
MessageBox.Show(
text: "このアプリケーションは既に起動中です。",
caption: "Error",
buttons: MessageBoxButtons.OK,
icon: MessageBoxIcon.Error);
return;
}
Application.Run(new SampleForm());
}
finally
{
if (createdNew)
{
mutex.ReleaseMutex();
}
mutex.Close();
}
}
private static void Application_ThreadException(object sender, ThreadExceptionEventArgs e)
{
var text = new StringBuilder();
text.AppendLine(e.Exception.Message);
text.AppendLine("スタックトレース:");
text.AppendLine(e.Exception.StackTrace);
MessageBox.Show(
text: text.ToString(),
caption: "Error",
buttons: MessageBoxButtons.OK,
icon: MessageBoxIcon.Error);
}
}
}
動作確認
動作確認用の Form を作成したので、テストしていきます。
二重起動防止については省略です。
using System;
using System.Threading.Tasks;
using System.Windows.Forms;
namespace App
{
public partial class SampleForm : Form
{
public SampleForm() => InitializeComponent();
// UI スレッドで例外
private void ThrowExceptionButton_Click(object sender, EventArgs e)
=> int.Parse(null);
// async/await で例外
private async void AwaitTaskRunButton_Click(object sender, EventArgs e)
=> await Task.Run(() => int.Parse(null));
// Task.Wait() で例外
private void TaskRunWaitButton_Click(object sender, EventArgs e)
=> Task.Run(() => int.Parse("Hoge")).Wait();
// Task.Run() で例外
private void TaskRunButton_Click(object sender, EventArgs e)
=> Task.Run(() => int.Parse("2147483648"));
}
}
1. UIスレッドでの例外発生
2. サブスレッドでの例外発生 (await)
3. サブスレッドでの例外発生 (Wait())
これは良い感じのように見えて、例外メッセージがおかしいですね。
「1 つ以上のエラーが発生しました。」と表示されており、どの例外を捕捉したのか分かりません。
これは Task.Wait() を使用した場合、複数の例外が AggregateException にまとめられる仕様のためです。
詳細は下記の記事をご覧ください(丸投げ①)。
個人的にはそもそも開発時に Task.Wait() や Task.Result の使用を非推奨(というか禁止)にしているため、今回このケースには対応していません...!
非推奨の理由は下記の MSDN 参照です(丸投げ②)。
async/await を使ってください...!
4. サブスレッドでの例外発生 (待たない場合)
このケースでは、そもそも例外を捕捉できません。
待たない場合は下記のように try catch で囲んでも捕捉できません...
private void TaskRunButton_Click(object sender, EventArgs e)
{
try
{
Task.Run(() => int.Parse("2147483648"));
}
catch (Exception ex)
{
// ※ここには入ってこない...
MessageBox.Show(ex.ToString());
}
}
おわりに
コピペ用にサクッと書き切るつもりが、非同期処理の例外関連で分量が多くなりました。
調べ物が少し大変でしたが、勉強になったのでよかったです...!