如果没有GetAwaiter方法,我可以等待吗?

时间:2014-09-25 19:41:22

标签: c# .net async-await system.reactive

我看到一些关于设计自定义等待类型的文章:

http://books.google.com.br/books?id=1On1glEbTfIC&pg=PA83&lpg=PA83&dq=create+a+custom+awaitable+type

现在考虑以下示例:

<Button x:Name="BtnA"
        Width="75"
        Margin="171,128,0,0"
        HorizontalAlignment="Left"
        VerticalAlignment="Top"
        Click="BtnA_Click"
        Content="Button A" />
<Button x:Name="BtnB"
        Width="75"
        Margin="273,128,0,0"
        HorizontalAlignment="Left"
        VerticalAlignment="Top"
        Content="Button B"  Click="BtnB_OnClick" />

private async void BtnA_Click(object sender, RoutedEventArgs e)
{
    MessageBox.Show("Awaiting Button B..");

    var sx = Observable.FromEvent<MouseButtonEventHandler, 
                                  MouseButtonEventArgs>(a => (b, c) => a(c),
                                  add => BtnB.PreviewMouseDown += add,
                                  rem => BtnB.PreviewMouseDown -= rem)
       .Do(a => a.Handled = true)
       .Take(1);

    await sx;

    MessageBox.Show("Button B Pressed after Button A");
}

private void BtnB_OnClick(object sender, RoutedEventArgs e)
{
    MessageBox.Show("Button B Pressed Without Click in Button A");
}

为什么我await IObservable<T>(在这种情况下订阅完成时),如果没有GetAwaiter()方法?

是由编译器实现的吗?是否有可能实现await无需显式的方法(某些场景正在使用哪种场景)? 为什么没有像:

public interface ITask<out T>
{
    IAwaiter<T> GetAwaiter();
}

public interface IAwaiter<out T> : ICriticalNotifyCompletion
{
    bool IsCompleted { get; }
    T GetResult();
}

...或用于创建自定义等待的真实界面?

2 个答案:

答案 0 :(得分:20)

首先,

你不能await 某些没有GetAwaiter方法可以返回某些的{{1>}方法1}},GetResultOnCompleted并实现IsCompleted

那么,您如何才能INotifyCompletion await

在实例方法之上,编译器也接受IObservable<T>扩展方法。在这种情况下,Reactive Extensions GetAwaiter提供了该扩展程序:

Observable

例如,这就是我们如何使字符串等待(编译但显然赢得了实际工作):

public static AsyncSubject<TSource> GetAwaiter<TSource>(this IObservable<TSource> source);

public static Awaiter GetAwaiter(this string s)
{
    throw new NotImplementedException();
}
public abstract class Awaiter : INotifyCompletion
{
    public abstract bool IsCompleted { get; }
    public abstract void GetResult();
    public abstract void OnCompleted(Action continuation);
}

答案 1 :(得分:2)

如果不存在,您可以通过扩展方法提供自己的GetAwaiter实现。

http://blogs.msdn.com/b/pfxteam/archive/2011/01/13/10115642.aspx