我有一个我无法控制的事件,它为我提供了数据。 eventArgs看起来像这样:
class MyEventArg {
bool IsLastItem {get;}
Data DataItem {get;}
}
我使用Rx将此事件转换为IObservable。但是如果IsLastItem为真,我想完成observable。
有什么优雅的想法吗?一种方法是通过我可以控制的主题来管道数据,以便在条件发生时设置OnComplete事件......
答案 0 :(得分:9)
如果要包含最后一个元素,则可以将流与仅包含最后一个元素以及与TakeWhile
组合的常规流合并。
这是一个简单的控制台应用程序来证明它:
var subject = new List<string>
{
"test",
"last"
}.ToObservable();
var my = subject
.Where(x => x == "last").Take(1)
.Merge(subject.TakeWhile(x => x != "last"));
my.Subscribe(
o => Console.WriteLine("On Next: " + o),
() => Console.WriteLine("Completed"));
Console.ReadLine();
打印:
On Next: test
On Next: last
Completed
<强>更新强>
如果底层的Observable实际上没有完成,则有一个错误压制了OnCompleted消息。我更正了代码以确保调用OnCompleted
如果你想避免多次为冷观察者订阅底层序列,你可以像这样重构代码:
var my = subject.Publish(p => p
.Where(x => x == "last").Take(1)
.Merge(p.TakeWhile(x => x != "last")));
答案 1 :(得分:2)
你在找这样的东西吗?
IObservable<MyEventArg> result =
myEventArgObservable.TakeWhile(arg => !arg.IsLastItem);
答案 2 :(得分:2)
public static IObservable<TSource> TakeWhileInclusive<TSource>(
this IObservable<TSource> source, Func<TSource, bool> predicate)
{
return Observable
.Create<TSource>(o => source.Subscribe(x =>
{
o.OnNext(x);
if (!predicate(x))
o.OnCompleted();
},
o.OnError,
o.OnCompleted
));
}