[英]Reactive Observable.Create for async producer without Task.Run
[英]Reactive framework eliminate while loop in Observable.Create
我正在处理从网络传来的可观察字节流,我希望将其用于一层抽象。 该格式有两个字节,其中包含下一条消息的长度。 我想很好地使其适应于反应式框架。 到目前为止,我感觉还不太对,所以我想知道我可能错过了哪些技巧来消除while循环。
这是我想到的概念:
public static IObservable<Stream> GetToplevelStreams(IObservable<byte> byteStreamArg) {
return Observable.Create((IObserver<Stream>o)=>{
bool done = false;
var byteStream = byteStreamArg.Do(
b => { }, (ex) => { done = true; }, () => { done = true; });
while (!done)
{
var size = byteStream.Take(2).
Aggregate(0, (n, b) => (n << 8) + b).Single();
var buf = byteStream.Skip(2).Take(size);
var stream = new MemoryStream(buf.ToEnumerable().ToArray());
if (!done)
{
o.OnNext(stream);
}
}
return (() => {});
});
}
IObservable在这里有点奇怪-记住您要返回“流的未来列表”-实际上我只是返回Stream
,或者可能是IObservable<byte[]>
,其中每个数组代表一条消息。 甚至做得更好,并返回IObservable<ParsedMessage>
同样,您的While循环会使此动作与异步行为发生异常。 像这样的东西怎么样:
public static IObservable<System.IO.Stream> GetToplevelStreams(IObservable<byte> byteStream)
{
return Observable.Create((IObserver<System.IO.Stream> o) =>
{
int? size1=null;
int? size=null;
var buf = new MemoryStream();
var subscription = byteStream.Subscribe(v =>
{
if (!size1.HasValue)
{
size1 = ((int)v) << 8;
}
else if (!size.HasValue)
{
size = size1.Value + v;
}
else
{
buf.WriteByte(v);
}
if (size.HasValue && buf.Length == size)
{
buf.Position = 0;
o.OnNext(buf);
buf.SetLength(0);
size1 = null;
size = null;
}
}, (ex)=>o.OnError(ex), ()=>o.OnCompleted());
return () => subscription.Dispose();
});
}
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.