2016-04-14 1 views
0

Я пытаюсь реализовать какой-то потоковый парсер. Предположим, что у меня есть поток целых чисел, и я объединяю их для создания нового объекта, который объединяет часть потока.Объединение нескольких элементов из наблюдаемых в новые объекты с использованием RxJava

Например, объект «сделан», когда целое число отрицательное. Чтобы это было просто, произведенные элементы будут строками чисел.

Вот простой пример:

Source: 1, -2, 3, 4, -5, 6, -7, 8, 9, 10, -11 
Output: "1-2", "343-5", "6-7", "8910-11" 

Я не могу найти правильное сочетание существующих операторов. Мое решение было обеспечить пользовательский оператор, который агрегирует данные до «кусок» является действительным, и излучает новый объект:

rx.Observable<Integer> src = rx.Observable 
    .from(Arrays.asList(1, -2, 3, 4, -5, 6, -7, 8, 9, 10, -11)); 

rx.Observable<String> res = src 
    .lift(new rx.Observable.Operator<String, Integer>() { 

     @Override 
     public rx.Subscriber<? super Integer> call(
      final rx.Subscriber<? super String> subscriber) { 

      return new rx.Subscriber<Integer>(subscriber) { 

       private final StringBuilder cur = new StringBuilder(); 

       @Override 
       public void onCompleted() { 
        if (!subscriber.isUnsubscribed()) { 
         subscriber.onCompleted(); 
        } 
       } 

       @Override 
       public void onError(Throwable e) { 
        if (!subscriber.isUnsubscribed()) { 
         subscriber.onError(e); 
        } 
       } 

       @Override 
       public void onNext(Integer i) { 
        if (subscriber.isUnsubscribed()) 
         return; 

        cur.append(i).append(''); 
        if (i < 0) { 
         subscriber.onNext(cur.toString()); 
         cur.delete(0, cur.length()); 
        } 
       } 
      }; 
     } 
    }); 

rx.observers.TestSubscriber<String> sub = 
    new rx.observers.TestSubscriber<>(); 
res.subscribe(sub); 
List<String> l1 = sub.getOnNextEvents(); 

Он работает хорошо, пока я не использовать некоторые позиционные операторы на них, как дубля (N) или combinationLatest.

rx.Observable<String> res2 = res.take(2); 

В этой ситуации я не получаю OnCompleted или OnError и меньшее количество элементов выбрасываются.

Похоже, каждый вход onNext должны иметь выход onNext называется.

Любые идеи?

ответ

0

Хм. Написание длинного сообщения помогло прийти с новой идеей, и оно работает. (Я стучу головой на стену уже 2 дня).

Производит Observer<Observer<String>> и испускает Observer.empty() до объекта готов. flatMap после.

Вот рабочий пример:

rx.Observable<rx.Observable<String>> res = src 
    .lift(new rx.Observable.Operator<rx.Observable<String>, Integer>() { 

     @Override 
     public rx.Subscriber<? super Integer> call(
      final rx.Subscriber<? super rx.Observable<String>> subscriber) { 

      return new rx.Subscriber<Integer>(subscriber) { 

       private final StringBuilder cur = new StringBuilder(); 

       // onError, onCompleted skipped 

       @Override 
       public void onNext(Integer i) { 
        rx.Observable<String> out; 
        cur.append(i); 
        if (i < 0) { 
         out = rx.Observable.just(cur.toString()); 
         cur.delete(0, cur.length()); 
        } else { 
         out = rx.Observable.<String>empty(); 
        } 
        subscriber.onNext(out); 
       } 
      }; 
     } 
    }); 

rx.Observable<String> res2 = res 
    .flatMap(stringObservable -> stringObservable) 
    .take(2); 
Смежные вопросы