[解決済み] await Task.Run(); return;" と "return Task.Run()" の違いは何ですか?
質問
以下の2つのコードの間に、何かコンセプトの違いがありますか?
async Task TestAsync()
{
await Task.Run(() => DoSomeWork());
}
と
Task TestAsync()
{
return Task.Run(() => DoSomeWork());
}
生成されたコードにも違いがあるのでしょうか?
EDITです。
との混同を避けるため
Task.Run
との混同を避けるため、同様のケースを
async Task TestAsync()
{
await Task.Delay(1000);
}
と
Task TestAsync()
{
return Task.Delay(1000);
}
遅れての更新です。
受理された答えに加えて、どのように
LocalCallContext
が処理されます。
CallContext.LogicalGetData は、非同期がない場合でも復元されます。なぜですか?
どのように解決するのですか?
大きな違いの1つは
例外の伝搬にあります。
例外は
async Task
メソッド内で投げられた例外は、返された
Task
オブジェクトに格納され、そのタスクが
await task
,
task.Wait()
,
task.Result
または
task.GetAwaiter().GetResult()
. から投げられた場合でも、この方法で伝搬されます。
同期
の部分から投げられたとしても、このように伝播します。
async
メソッドを使用します。
次のコードを考えてみましょう。
OneTestAsync
と
AnotherTestAsync
は全く異なる挙動をします。
static async Task OneTestAsync(int n)
{
await Task.Delay(n);
}
static Task AnotherTestAsync(int n)
{
return Task.Delay(n);
}
// call DoTestAsync with either OneTestAsync or AnotherTestAsync as whatTest
static void DoTestAsync(Func<int, Task> whatTest, int n)
{
Task task = null;
try
{
// start the task
task = whatTest(n);
// do some other stuff,
// while the task is pending
Console.Write("Press enter to continue");
Console.ReadLine();
task.Wait();
}
catch (Exception ex)
{
Console.Write("Error: " + ex.Message);
}
}
もし私が
DoTestAsync(OneTestAsync, -2)
を呼び出すと、次のような出力が得られます。
続行するにはEnterキーを押してください エラーです。1つ以上のエラーが発生しました。 エラー: 2回目
注、私は Enter を押さないと表示されません。
さて、もし私が
DoTestAsync(AnotherTestAsync, -2)
を呼び出すと、その中のコード・ワークフローは
DoTestAsync
の中のコードのワークフローはかなり異なっており、出力も異なっています。今回、私は
Enter
:
エラーです。値は-1(無限のタイムアウトを意味する)、0、または正の整数のいずれかを指定する必要があります。 パラメータ名:millisecondsDelayError: 1位
どちらの場合も
Task.Delay(-2)
はそのパラメータを検証している間、冒頭で投げます。これはでっち上げのシナリオかもしれませんが、理論的には
Task.Delay(1000)
も投げるかもしれません。例えば、基礎となるシステムタイマー API が失敗した場合です。
余談ですが、エラー伝搬のロジックはまだ
async void
メソッド
(対して
async Task
メソッドに対して)。の中で発生した例外は
async void
メソッド内で発生した例外は、現在のスレッドの同期コンテキストで (SynchronizationContext.Post
を通して)、現在のスレッドが持っている場合(
SynchronizationContext.Current != null)
. そうでなければ、それは
ThreadPool.QueueUserWorkItem
). 呼び出し元は同じスタックフレーム上でこの例外を処理する機会はありません。
TPLの例外処理の動作について、もう少し詳細を投稿しました。 はこちら と はこちら .
Q
: の例外伝播の挙動を模倣することは可能でしょうか?
async
メソッドを非同期
Task
-ベースのメソッドで、後者が同じスタックフレームで投げないようにするためですか?
A : 本当に必要なら、そう、そのためのトリックがあるのです。
// async
async Task<int> MethodAsync(int arg)
{
if (arg < 0)
throw new ArgumentException("arg");
// ...
return 42 + arg;
}
// non-async
Task<int> MethodAsync(int arg)
{
var task = new Task<int>(() =>
{
if (arg < 0)
throw new ArgumentException("arg");
// ...
return 42 + arg;
});
task.RunSynchronously(TaskScheduler.Default);
return task;
}
ただし
は特定の条件下で
(がスタック上で深すぎる場合など)。
RunSynchronously
はまだ非同期で実行される可能性があります。
もう一つの顕著な違いは は
async
/
await
バージョンはデフォルトでない同期コンテキストでデッドロックを起こしやすくなっています。
. 例えば、以下はWinFormsやWPFのアプリケーションでデッドロックが発生します。
static async Task TestAsync()
{
await Task.Delay(1000);
}
void Form_Load(object sender, EventArgs e)
{
TestAsync().Wait(); // dead-lock here
}
非同期バージョンに変更すれば、デッドロックは発生しません。
Task TestAsync()
{
return Task.Delay(1000);
}
デッドロックの性質については、Stephen Cleary が彼の ブログ .
関連
-
[解決済み] C#のStringとstringの違いは何ですか?
-
[解決済み] C#のconstとreadonlyの違いは何ですか?
-
[解決済み] フィールドとプロパティの違いは何ですか?
-
[解決済み] 2つの日付の差(日数)を計算する?
-
[解決済み] async」と「await」の使い方とタイミング
-
[解決済み] SelectとSelectManyの違い
-
[解決済み] キーワード「ref」と「out」の違いは何ですか?
-
[解決済み] C#の==とEquals()の違いについて
-
[解決済み] Task.Runの正しい使い方とasync-awaitだけの使い方
-
[解決済み】await Promise.all()と複数awaitの違いって何?
最新
-
nginxです。[emerg] 0.0.0.0:80 への bind() に失敗しました (98: アドレスは既に使用中です)
-
htmlページでギリシャ文字を使うには
-
ピュアhtml+cssでの要素読み込み効果
-
純粋なhtml + cssで五輪を実現するサンプルコード
-
ナビゲーションバー・ドロップダウンメニューのHTML+CSSサンプルコード
-
タイピング効果を実現するピュアhtml+css
-
htmlの選択ボックスのプレースホルダー作成に関する質問
-
html css3 伸縮しない 画像表示効果
-
トップナビゲーションバーメニュー作成用HTML+CSS
-
html+css 実装 サイバーパンク風ボタン
おすすめ
-
[解決済み] [Solved] 1つ以上のエンティティで検証に失敗しました。詳細は'EntityValidationErrors'プロパティを参照してください [重複]。
-
[解決済み】C#で四捨五入する方法
-
[解決済み】プロジェクトビルド時のエラー。エディタでスクリプトにコンパイルエラーがあるため、Playerのビルドにエラーが発生する
-
[解決済み] DBNullから他の型にオブジェクトをキャストすることができない
-
[解決済み] 'IEnumerable<SelectListItem>' 型の ViewData アイテムで、キーが国であるものは存在しない。
-
[解決済み】なぜこのコードはInvalidOperationExceptionを投げるのですか?
-
[解決済み】Socket.Selectがエラー "An operation was attempted on something that is not a socket" を返す。
-
[解決済み】Moqを使用してメソッド呼び出しを検証する
-
[解決済み】ランダムなブーリアンを生成する最速の方法
-
[解決済み】インデックスが範囲外でした。コレクションパラメータname:indexのサイズより小さく、非負でなければなりません。