web-dev-qa-db-ja.com

簡単な非同期メソッドの書き方は?

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が指す行を選択しました)

17
ghord

Mainを非同期としてマークできるかどうかはわかりませんが、asyncを使用するメソッドの宣言にawaitキーワードを含める必要があります。例えば:

public async void DoStuffAsync ()
{
    var myClass = new MyClass ();

    int result = await myClass.TestAsync ();
}
8
Bodrick

非同期メソッドの戻り値の型は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"]);
        }

    }
4
Andrew Scoppa

awaitWait()と同じではありません。 awaitを実行することは、そのメソッドを大幅に書き直すことであり、特に、そのメソッドが呼び出し元にどのように終了するかについての期待に影響を与えます。コンパイラにいくつかのことを有効にするように指示する以外は、実際には do 多くはありません(警告:戻り値の型)(unsafecheckeduncheckedなどのスイッチも同様です)それについて)-しかし考えてみてください:これは実際に重要ですあなたの例では非常にMain()が終了した場合(そして他のスレッドがないと仮定した場合)-exe is toast 。なくなった。もはや存在しない。 asyncを追加すると、メソッド exits が、 finish を持っているとは限らないと考えるようになります。準備が整う前にMain()を終了させたくはありません。

二次的な効果として、このスイッチは、メソッドがTaskのようなものしか返すことができないことも形式化します。スイッチがないと、後で非同期にしたくなるかもしれません。これは、大幅な変更になる可能性があります。

4
Marc Gravell