我看到一些关于设计自定义等待类型的文章:
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();
}
...或用于创建自定义等待的真实界面?
答案 0 :(得分:20)
你不能await
某些没有GetAwaiter
方法可以返回某些的{{1>}方法1}},GetResult
,OnCompleted
并实现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