Async/awaitキーワードを指定した最新のCTP5を使用して、コンパイルできないコードをいくつか作成しました。
class Program
{
public class MyClass
{
async public Task<int> Test()
{
var result = await TaskEx.Run(() =>
{
Thread.Sleep(3000);
return 3;
});
return result;
}
}
static void Main(string[] args)
{
var myClass = new MyClass();
//The 'await' operator can only be used in a method or lambda marked with the 'async' modifier error ??!!
int result = await myClass.Test();
Console.ReadLine();
}
}
「 'await'演算子は、 'async'修飾子エラーでマークされたメソッドまたはラムダでのみ使用できます」の理由は何ですか? (Visual Studioが指す行を選択しました)
Mainを非同期としてマークできるかどうかはわかりませんが、async
を使用するメソッドの宣言にawait
キーワードを含める必要があります。例えば:
public async void DoStuffAsync ()
{
var myClass = new MyClass ();
int result = await myClass.TestAsync ();
}
非同期メソッドの戻り値の型はvoidまたはTaskです。戻り値の型が無効でない場合でも、呼び出し元は、.Net 4で導入されたメインエントリメソッド内の標準の待機メカニズムを使用できます(非同期としてマークすることはできません)。簡単な例を次に示します。
static void Main(string[] args)
{
string address = "http://api.worldbank.org/countries?format=json";
Task t = LoadJsonAsync(address);
// do other work while loading
t.Wait();
Console.WriteLine("Hit ENTER to exit...");
Console.ReadLine();
}
private async static Task LoadJsonAsync(string address)
{
HttpClient client = new HttpClient();
HttpResponseMessage response = await client.GetAsync(address);
// Check that response was successful or throw exception
response.EnsureSuccessStatusCode();
// Read response asynchronously as JsonValue and write out top facts for each country
JsonArray readTask = await response.Content.ReadAsAsync<JsonArray>();
Console.WriteLine("First 50 countries listed by The World Bank...");
foreach (var country in readTask[1])
{
Console.WriteLine(" {0}, Capital: {1}",
country.Value["name"],
country.Value["capitalCity"]);
}
}
await
はWait()
と同じではありません。 await
を実行することは、そのメソッドを大幅に書き直すことであり、特に、そのメソッドが呼び出し元にどのように終了するかについての期待に影響を与えます。コンパイラにいくつかのことを有効にするように指示する以外は、実際には do 多くはありません(警告:戻り値の型)(unsafe
、checked
、unchecked
などのスイッチも同様です)それについて)-しかし考えてみてください:これは実際に重要ですあなたの例では非常に。 Main()
が終了した場合(そして他のスレッドがないと仮定した場合)-exe is toast 。なくなった。もはや存在しない。 async
を追加すると、メソッド exits が、 finish を持っているとは限らないと考えるようになります。準備が整う前にMain()
を終了させたくはありません。
二次的な効果として、このスイッチは、メソッドがTask
のようなものしか返すことができないことも形式化します。スイッチがないと、後で非同期にしたくなるかもしれません。これは、大幅な変更になる可能性があります。