質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.49%
C#

C#はマルチパラダイムプログラミング言語の1つで、命令形・宣言型・関数型・ジェネリック型・コンポーネント指向・オブジェクティブ指向のプログラミング開発すべてに対応しています。

非同期処理

非同期処理とは一部のコードを別々のスレッドで実行させる手法です。アプリケーションのパフォーマンスを向上させる目的でこの手法を用います。

.NET Framework

.NET Framework は、Microsoft Windowsのオペレーティングシステムのために開発されたソフトウェア開発環境/実行環境です。多くのプログラミング言語をサポートしています。

Q&A

解決済

1回答

8751閲覧

WinFormsのFormClosingイベントハンドラをasyncにすると画面が閉じません

ry18847

総合スコア7

C#

C#はマルチパラダイムプログラミング言語の1つで、命令形・宣言型・関数型・ジェネリック型・コンポーネント指向・オブジェクティブ指向のプログラミング開発すべてに対応しています。

非同期処理

非同期処理とは一部のコードを別々のスレッドで実行させる手法です。アプリケーションのパフォーマンスを向上させる目的でこの手法を用います。

.NET Framework

.NET Framework は、Microsoft Windowsのオペレーティングシステムのために開発されたソフトウェア開発環境/実行環境です。多くのプログラミング言語をサポートしています。

0グッド

0クリップ

投稿2018/01/09 02:56

asyncなFormClosingイベントハンドラで、以下の実装をしたところFormClosedイベントが起きず、フォームが閉じません。
ブレークポイントを置いて実行したところ、e.Cancel = false; は実行されるのですが、なぜ画面が閉じないのでしょうか?

###該当のソースコード

private async void Form1_FormClosing(object sender, FormClosingEventArgs e) { e.Cancel = true; await Task.Delay(1000); //重たい処理 e.Cancel = false; }

###試したこと
以下のコードの場合、閉じたり閉じなかったりします。
コントロールボックスの×ボタンを連打するとそのうち閉じるのですが、なぜこんな動作をするんでしょうか?
(Taskが即座に終わるかどうかが関係している?)

private async void Form1_FormClosing(object sender, FormClosingEventArgs e) { e.Cancel = true; await Task.Run(() => { return; }); e.Cancel = false; }

###補足情報(言語/FW/ツール等のバージョンなど)
.Net Framework 4.5
VS2012

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ベストアンサー

await で一度イベントハンドラが終了します。その時 e.Canceltrue です。
ですのでフォームは閉じません。

しかし e.Cancel が評価される前にタスクが終了して e.Cancel が書き換えられるとフォームは閉じます。

追記

わかりにくかったかもしれないので順を追って説明します。

これは System.Windows.Forms.Form のソースです。
Form.cs

C#

1OnFormClosing(e); 2if (e.Cancel) 3{ 4 dialogResult = DialogResult.None; 5} 6else 7{ 8 // we have called closing here, and it wasn't cancelled, so we're expecting a close 9 // call again soon. 10 CalledClosing = true; 11}

上記の OnFormClosing(e);FormClosing イベントハンドラを(あれば)呼びます。
ユーザーはハンドラの中で e.Cancel を設定します。
イベントハンドラが次のようになっている場合

C#

1private async void Form1_FormClosing(object sender, FormClosingEventArgs e) 2{ 3 e.Cancel = true; 4 await Task.Run(() => 5 { 6 return; 7 }); 8 e.Cancel = false; 9}

await によって一度ハンドラが終了するので、次のようなハンドラを指定したのと同じことになります。

C#

1private async void Form1_FormClosing(object sender, FormClosingEventArgs e) 2{ 3 e.Cancel = true; 4}

そして同時に次の非同期処理が登録されます。
(実際に実行されるのは同時とは限りません)

C#

1Task.Run(() => 2{ 3 return; 4}); 5e.Cancel = false;

この非同期処理は return のみのタスクを実行し、それが終了すると e.Cancel = false というタスクを実行しますので、整理すると次の内容とほぼ同じです。

C#

1Task.Run(() => 2{ 3 e.Cancel = false; 4});

さてこの非同期処理がいつ行われるかですが、それは実行してみるまでわかりません。
この処理によって e.Cancel が書き換えられるわけですが、それが Form.cs 中の

C#

1if (e.Cancel)

この部分より前に行われると、e.Cancelfalse となって else ブロックが実行されます。
また、後に行われたならば真のブロックが実行されます。

重たい処理の場合はおそらく間に合わないので e.Canceltrue となります。
二番目の例の場合は間に合うか間に合わないかは状況次第です。

以上が処理の流れとなります。

投稿2018/01/09 03:25

編集2018/01/09 04:14
Zuishin

総合スコア28660

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

Zuishin

2018/01/09 03:34

await はその名前から「待つ」と理解しがちですが、「非同期タスクが終わった後の処理を予約して終了する」と考えた方が理解しやすいです。
ry18847

2018/01/09 04:20

回答ありがとうございます。 FormClosingイベントハンドラの処理がすべて完了してからFormClosedが起きると思っていましたが、違うのですね。 ということは、FormClosingイベントで非同期に重い処理をしたい場合は、重たい処理以降で必要に応じてthis.Close();し直す(この場合は重たい処理が再度走らないようにフラグ管理等する)必要があるという認識でいいでしょうか? (そもそも重い処理を終わらせてからthis.Close(); したほうがいいのでしょうけど・・・)
Zuishin

2018/01/09 04:22

フォームが閉じるのに必要な処理なら非同期にせず同期処理で待たせましょう。 その間のループ中に `Application.DoEvents()` を繰り返し呼んでメッセージを処理するといいと思います。
ry18847

2018/01/09 04:54

ご丁寧な解説をいただき、ありがとうございます。 同期で待たせてApplication.DoEvents()で処理することを検討してみたいと思います。 最後に1点、今回例にした重い処理の正体は画面を閉じるとき以外にも実行する非同期通信処理メソッドでして、Closingイベント中ではループがありません。 この場合はどのようにDoEvents()を繰り返し実行させればよいでしょうか? (Application.DoEvents();を無限ループ処理するTaskを作ってWhenAny()を使う?)
Zuishin

2018/01/09 04:59 編集

その方法が簡単だと思います。 どの程度重いのかにもよりますが、ShowDialog() でプログレスダイアログを出してその Shown イベントで非同期処理を行い、処理が終わってダイアログが閉じられるのを待つという手もあります。
ry18847

2018/01/09 05:03

ご回答ありがとうございました! イベントハンドラおよび非同期処理については引き続き勉強していきたいと思います。
Zuishin

2018/01/09 05:19

書くのを忘れていました。無限ループを使う場合には非同期処理が始まらない場合があるので、その時には非同期処理を別スレッドに追い出す必要があります。TaskScheduler を調べてください。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.49%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問