スキップしてメイン コンテンツに移動

[C#] 非同期処理(TASK)の例外処理を極めろ!

enter image description here

C#のTaskの例外処理ちゃんとやってますか??

**C#**で非同期処理を実装する場合、Taskクラスを使っている方が多いと思います。Taskクラスを使えば、以下のように、たった数行のコードで簡単に非同期処理を作る事が出来ます。

Task.Run(()  =>  Console.WriteLine("何やら重たい処理..."));

しかし、このコードには重大な問題があります。
例外処理がまったく出来てません。

Task内で発生した例外は、適切に処理しないと、例外がキャッチされずに正常終了してしまいます。

「それでもいい!」と言うのであれば別ですが、通常は例外が発生したら、ユーザに通知 or ログ出力等を行うと思います。

今回は、非同期処理内で発生した例外を、適切に処理する方法について紹介したいと思います。

Task で発生した例外の処理方法

1. Wait() で Taskの終了を待機する場合

Task.Wait() で Taskの終了を待機する場合、Wait() 関数の部分を try ~ catch で囲めば、Task内で発生した例外を拾う事が出来ます。


//例外を発生させる Task を作成  
Task task = new Task(() => {  
    throw new Exception("エラーが発生しました!!");  
});  
task.Start();  
  
// Wait()でタスクの終了を待つ  
try  {  
    task.Wait();  
} catch (AggregateException e)  {  
    //タスク内で発生した例外は、ここでキャッチできる  
    Console.WriteLine(e.InnerException.Message);  
}
POINT Task内で発生した例外は、AggregateExceptionにラップされて、親スレッドにスローされます。
その為、実際に発生した例外を取得するには、AggregateException.InnerExceptionプロパティより発生した例外を取得します。

2. async / await を使用する場合

async / await パターンの場合、通常の例外と同じように、try ~ catch で非同期の例外処理を行う事ができます。
AggregateExceptionで例外がラップされる事もない為、ホントに普通に書けます。
正直、こんなに簡潔に非同期の例外処理が書けるなんて、衝撃でした。

public async Task Sample() {

    try {
        //例外を発生させる Task を作成
        await Task.Run(() => {
            throw new ApplicationException("エラーが発生しました!!");
        });
    } catch (ApplicationException e) {
        Console.WriteLine(e.Message);
    }
}

3. 突き放し型の場合

これまでとは違い、Taskの終了を待機しない突き放し型の場合についてです。
この場合Task.ContinueWith()関数を使用して、下のコードの様に、Task終了時に例外が発生しているかチェックします。

//例外を発生させる Task を作成
Task task = new Task(() => {
    throw new Exception("エラーが発生しました!!");
});
task.Start();

//タスク完了時の処理
task.ContinueWith((t) => {
    if (t.Exception == null) {
        //通常の処理
    } else {
        //タスク内でエラーが発生した時の処理
        Console.WriteLine(t.Exception.InnerException);
    }
});

まとめ

Task の非同期処理内で発生した例外は、簡単なコードで検知する事ができます。
正しく処理を行って、障害をすばやく検知できるシステムを作りたいですね。。。

コメント

このブログの人気の投稿

axiosの使い方まとめ (GET/POST/例外処理)

axiosの使い方まとめ (GET/POST/例外処理)最近何かとよく使うJavaScriptでAJAX通信を行うaxiosについて、簡単に使い方をまとめました。GETリクエストをaxiosで送るまずはGETリクエストをaxiosで送る方法です。const res =await axios.get('/users') console.log(res.data)分割代入の記法を使うと、以下のようにも書けますconst{data}=await axios.get('/users') console.log(data)クエリパラメータ (URLパラメータ)を指定クエリパラメータを指定する方法は2つあります。1つ目は、axios.getに指定するURLに直接記述する方法です。axios.get('/user?id=123')2つめは、axios.getの第2引数に、オプション指定する方法です。axios.get('/user',{ params:{ id:123}})POSTリクエストをaxiosで送る次はPOSTリクエストをaxiosで送る方法です。JSON形式でPOSTするJSON形式でPOSTする場合は、axios.postの第2引数に、送信するデータをJavaScriptオブジェクトで指定します。const res =await axios.post('/user',{ id:123, name:'Yamada Tarou'})application/x-www-form-urlencoded形式でPOSTするapplication/x-www-form-urlencoded形式でPOSTする場合は、URLSearchParamsを使います。var params =newURLSearchParams() params.append('id',123) params.append('name','Yamada Tarou')const res =await axios.post('/user', params)スポンサーリンク axios でファイルをアップロードする画像などのファイルを、…

[VB, C#] Windows 8, Window 10 で ImeModeが制御できない問題を解決する

[VB, C#] Windows 8, Window 10 で ImeModeが制御できない問題を解決するタイトルの通りですが、Windows 8 以降では Windows Form アプリケーションで、コントロールの ImeMode に Katakana や KatakanaHalf を設定しても、カタカナになってくれません。なぜ ImeMode が効かないのか?Windows 8 以降、IME Mode の切り替えは、ユーザー単位で切り替わるようになった為、アプリから IME Mode 制御が出来ないようになりました。
(IME をON にした場合、常に ひらがな モードになます)※ Windows 7までは、IME Modeの切り替えはアプリ単位で行われていた為、問題なくアプリから IME制御が行えました。スポンサーリンク 対処方法Windows 8 以降、IMEの制御は、InputScope クラスの利用が推奨されています。
しかし、InputScope クラスは、WPF、Windows ストアアプリでしか使えない為、Windows Formアプリでは使用できません。
(Windows Form はもう使うな!という事でしょうか (涙) )結論としては、コントールパネルの設定で、IMEの制御をユーザ単位から アプリ単位に変更する事ができます。
これで、Windows Formアプリでも 従来通りIMEの制御を行う事が出来ます。おわりにこの方法だと、アプリをインストールする端末すべてに設定が必要となり、とっても面倒です。。。
しかし、今の所これしか方法がない状態です。
これからは Windows Formではなく、WPFや Windows ストアアプリで作れという事ですかね (^^;)

MailKitの使い方! エンコーディング指定や添付ファイをメールで送信する方法[C#/VB Tips]

MailKitの使い方! エンコーディング指定や添付ファイをメールで送信する方法[C#/VB Tips]MailKitを使ってメールを送るサンプルコードです。(C#)UTF8/iso-2022-jpのエンコーディング指定、GMail/YahooのSMTPサーバで送るなど、4つのサンプルコードでMailKitの使い方を紹介します。MailKitって何?2017年に.NET標準のSystem.Net.Mail.SmtpClientが廃止予定となり、Microsoftより今後はオープンソースライブラリである、MailKitに置き換えるとアナウンスがありました。既にSmtpClientは非推奨になっており、今後は廃止されていきます。現在、SmtpClientを使用したソースコードには、Visual StudioからMailKitを使うよう警告が出るようになっています。さっそく、MailKitを使ってメールを送信するサンプルコードを作っていきます。UTF8でメールを送信文字エンコーディングを、UTF8でメールを送信するサンプルコードです。
MailKitは、デフォルトの文字エンコーディングがUTF8なっている為、シンプルなコードでメールを送信する事ができます。var host ="<smtp server name>"; var port =25;// or 587using(var smtp =new MailKit.Net.Smtp.SmtpClient()){//SMTPサーバに接続する smtp.Connect(host, port, MailKit.Security.SecureSocketOptions.Auto);//認証が必要な場合は、以下のコメントを解除//smtp.Authenticate("<id>", "<password>");//送信するメールを作成する var mail =new MimeKit.MimeMessage(); var builder =new MimeKit.BodyBuilder(); mail.From.Add(new MimeKit.MailboxAddress("",&quo…