【发布时间】:2017-10-02 11:11:43
【问题描述】:
我想将我遗留的基于事件的方法转换为基于可观察的方法,但我对 Rx 很陌生,所以我现在卡住了。
我有一个事件源,现在是可观察的。在某个时间点,我必须启动一个方法,该方法通过返回该行中的下一个元素或如果超时则返回 null。
基于事件的方法如下所示:
public async Task<ReaderEvent> WaitForReaderAsync(int PlaceId, TimeSpan waitFor)
{
ReaderEvent result = null;
using (var cts = CancellationTokenSource.CreateLinkedTokenSource(new [] { topLevelToken }))
{
cts.CancelAfter(waitFor);
EventHandler<ReaderEvent> localHandler = (o, e) =>
{
if (e.PlaceId == PlaceId)
{
result = e;
cts.Cancel();
}
};
ReaderEventHandler += localHandler;
try
{
await Task.Delay(waitFor, cts.Token).ConfigureAwait(false);
}
catch (OperationCanceledException) { }
catch (Exception ex)
{
//...
}
ReaderEventHandler -= localHandler;
}
return result;
}
如您所见,这个想法是延迟被我等待的事件的到来取消,或者令牌源在特定时间后被配置取消。很干净。
现在,Rx 版本:
public async Task<ReaderEvent> WaitForReaderAsync(int PlaceId, TimeSpan waitFor)
{
ReaderEvent result = null;
var observable = _OnReaderEvent.FirstAsync(r => r.PlaceId == PlaceId);
using (var cts = CancellationTokenSource.CreateLinkedTokenSource(new [] { topLevelToken }))
{
cts.CancelAfter(waitFor);
using (observable.Subscribe(x => {
result = x;
cts.Cancel();
{
try
{
await Task.Delay(waitFor, cts.Token).ConfigureAwait(false);
}
catch (OperationCanceledException) { }
}
}
return result;
}
不是那么干净……更糟糕的是…… 我也尝试过超时扩展。但由于这是一次性订阅,我仍然需要在处理订阅之前以某种方式等待。唯一的区别是 OnError 会取消本地令牌,而不是 CancelAfter 的内置机制。
是否有任何击球手/更简洁(更依赖 Rx)的方式来做到这一点?
谢谢!
【问题讨论】:
标签: c# task-parallel-library system.reactive