将具有不规则长度字节数组的IObservable <byte []>转换为具有常规长度数组的IObservable <byte []> </byte []> </byte []>

时间:2011-10-07 14:18:53

标签: c# system.reactive

我有一个IObservable<byte[]>,它在字节数组中给出了不确定的字节数。我想知道如何从那里开始,在每个字节数组中返回一个具有设定字节数的IObservable<byte[]>。假设我们一次想要10个字节。

也就是说,如果我要订阅,我会得到以下输入:

{1, 2, 3, 4}
{5, 6}
{7, 8, 9}
{10}
{11, 12, 13, 14, 15}
{16}
{17, 18}
{19, 20}

Bytes.Subscribe(b => Console.WriteLine(b.Length));

输出为

3
2
3
1
5
1
2
2

我想要的是将上面的输入转换为:

{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}
{11, 12, 13, 14, 15, 16, 17, 18, 19, 20}

Bytes.<WhateverItTakesToDoThat>.Subscribe(b => Console.WriteLine(b.Length));

输出为

10
10

如果进入的字节数大于单个输出数据包,它也必须工作,即:

{21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32}
{33, 34, 35, 36, 37, 38, 39, 40, 41}
{42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52}

应该变成

{21, 22, 23, 24, 25, 26, 27, 28, 29, 30}
{31, 32, 33, 34, 35, 36, 37, 38, 39, 40}
{41, 42, 43, 44, 45, 46, 47, 48, 49, 50}

(并坚持{51,52},等待更多的投入来)

2 个答案:

答案 0 :(得分:4)

这很容易。试试这个:

    Bytes
        .SelectMany(b => b)
        .Buffer(10)
        .Select(bs => bs.ToArray());

答案 1 :(得分:1)

经过一番思考和修修后我想出了一个解决方案。以下代码执行我想要的操作:

Bytes.Select( b => b.ToObservable() ) // Convert input to IObservable<IObservable<byte>>
.Merge( 1 ) // Merges the IObservable<IObservable<byte>> to an IObservable<byte>
            // with the bytes in the right order
.Buffer( 4 ) // Wait until we have 4 bytes ready
.Select( bl => bl.ToArray() ) // Take these 4 bytes and turn them back into an array
.Subscribe( b => Console.WriteLine( b.Length ) );

这可能是效率低下的,我几乎可以肯定,这不是他最有效的方法,所以如果有人能够提出更好,更有效的解决方案,我全都听见了!