我正在实现一个进程间消息传递系统,例如客户端可以从服务器请求某些数据。服务器需要能够以部分回复的形式发回其回复,并且还需要能够通知客户端以防发生任何异常。
目前,我通过 3 种消息类型执行此操作:
class PartialResponse : ResponseMessage { ... }
class ResponseError : ResponseMessage { ... }
class ResponseComplete : ResponseMessage { ... ]
例如,客户端请求数据,服务器发回 0-N PartialResponse 消息,后跟 ResponseError 或 ResponseComplete。
我正在使用的库(以 NetMQ 作为其传输层的 Obvs)会将所有可能的消息流公开为
IObservable<ResponseMessage>
虽然这个可观察的流永远不会完成,我相信也永远不会出错(除非可能是某些 Obvs/NetMQ 内部异常等)。
我想将其转换为 IObservable<PartialResponse>,它在原始流推送 ResponseComplete 消息时完成,并在遇到 ResponseError 消息或输入流中的实际错误时出错。例如:
IObservable<PartialResponse> transform(IObservable<ResponseMessage> input)
{
var subject = new Subject<PartialResponse>();
input.Subscribe(
x =>
{
if(x is PartialResponse r)
subject.OnNext(r);
else if(x is ResponseComplete)
subject.OnCompleted();
else if(x is ResponseError err)
subject.OnError(new Exception(err?.ToString()));
else
throw new InvalidOperationException();
},
ex =>
{
subject.OnError(ex);
}
);
return subject;
}
这段代码实际上应该可以工作,但可能很糟糕——尤其是因为它直接订阅了输入的可观察序列。
有没有更好/更清洁的方法来转换可观察序列?