RxJava:缓冲项目,直到当前项目的某些条件为真

时间:2016-03-09 01:21:08

标签: java rx-java

这是我想弄清楚的一个片段:

class RaceCondition {

    Subject<Integer, Integer> subject = PublishSubject.create();

    public void entryPoint(Integer data) {
        subject.onNext(data);
    }

    public void client() {
        subject /*some operations*/
                .buffer(getClosingSelector())
                .subscribe(/*handle results*/);
    }

    private Observable<Integer> getClosingSelector() {
        return subject /* some filtering */;
    }
}

有一个Subject接受来自外部的事件。有一个客户订阅了这个主题来处理事件,还有buffer个。这里的主要思想是每次都应该根据使用流中的项目计算的某些条件来发送缓冲的项目。

为此目的,缓冲区边界本身会侦听主题。

一个重要的期望行为:每当边界发出项目时,它也应该包含在buffer的以下发射中。在到达buffer之前,从关闭选择器发出的项目(至少是我认为的那样)不是当前配置的情况,因此它不包含在当前排放,但留下等待下一个。

有没有办法基本上让关闭选择器等待项目首先被缓冲?如果没有,是否有另一种方法来缓冲和释放基于下一个传入项目的项目?

1 个答案:

答案 0 :(得分:4)

如果我理解正确,你想要缓冲,直到一些谓词允许它基于项目。您可以使用一组复杂的运算符来完成此操作,但可能更容易编写自定义运算符:

public final class BufferUntil<T> 
implements Operator<List<T>, T>{

    final Func1<T, Boolean> boundaryPredicate;

    public BufferUntil(Func1<T, Boolean> boundaryPredicate) {
        this.boundaryPredicate = boundaryPredicate;
    }

    @Override
    public Subscriber<? super T> call(
            Subscriber<? super List<T>> child) {
        BufferWhileSubscriber parent = 
                new BufferWhileSubscriber(child);
        child.add(parent);
        return parent;
    }

    final class BufferWhileSubscriber extends Subscriber<T> {
        final Subscriber<? super List<T>> actual;

        List<T> buffer = new ArrayList<>();

        /**
         * @param actual
         */
        public BufferWhileSubscriber(
                Subscriber<? super List<T>> actual) {
            this.actual = actual;
        }

        @Override
        public void onNext(T t) {
            buffer.add(t);
            if (boundaryPredicate.call(t)) {
                actual.onNext(buffer);
                buffer = new ArrayList<>();
            }
        }

        @Override
        public void onError(Throwable e) {
            buffer = null;
            actual.onError(e);
        }

        @Override
        public void onCompleted() {
            List<T> b = buffer;
            buffer = null;
            if (!b.isEmpty()) {
                actual.onNext(b);
            }
            actual.onCompleted();
        }
    }
}