All Downloads are FREE. Search and download functionalities are using the official Maven repository.

rx.internal.operators.BlockingOperatorNext Maven / Gradle / Ivy

/**
 * Copyright 2014 Netflix, Inc.
 * 
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 * 
 * http://www.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package rx.internal.operators;

import java.util.Iterator;
import java.util.NoSuchElementException;
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.atomic.AtomicIntegerFieldUpdater;

import rx.Notification;
import rx.Observable;
import rx.Subscriber;
import rx.exceptions.Exceptions;

/**
 * Returns an Iterable that blocks until the Observable emits another item, then returns that item.
 * 

* */ public final class BlockingOperatorNext { /** * Returns an {@code Iterable} that blocks until the {@code Observable} emits another item, then returns * that item. * * @param items * the {@code Observable} to observe * @return an {@code Iterable} that behaves like a blocking version of {@code items} */ public static Iterable next(final Observable items) { return new Iterable() { @Override public Iterator iterator() { NextObserver nextObserver = new NextObserver(); final NextIterator nextIterator = new NextIterator(nextObserver); items.materialize().subscribe(nextObserver); return nextIterator; } }; } // test needs to access the observer.waiting flag non-blockingly. /* private */static final class NextIterator implements Iterator { private final NextObserver observer; private T next; private boolean hasNext = true; private boolean isNextConsumed = true; private Throwable error = null; private NextIterator(NextObserver observer) { this.observer = observer; } // in tests, set the waiting flag without blocking for the next value to // allow lockstepping instead of multi-threading /** * In tests, set the waiting flag without blocking for the next value to * allow lockstepping instead of multi-threading * @param value use 1 to enter into the waiting state */ void setWaiting(int value) { observer.setWaiting(value); } @Override public boolean hasNext() { if (error != null) { // If any error has already been thrown, throw it again. throw Exceptions.propagate(error); } // Since an iterator should not be used in different thread, // so we do not need any synchronization. if (hasNext == false) { // the iterator has reached the end. return false; } if (isNextConsumed == false) { // next has not been used yet. return true; } return moveToNext(); } private boolean moveToNext() { try { Notification nextNotification = observer.takeNext(); if (nextNotification.isOnNext()) { isNextConsumed = false; next = nextNotification.getValue(); return true; } // If an observable is completed or fails, // hasNext() always return false. hasNext = false; if (nextNotification.isOnCompleted()) { return false; } if (nextNotification.isOnError()) { error = nextNotification.getThrowable(); throw Exceptions.propagate(error); } throw new IllegalStateException("Should not reach here"); } catch (InterruptedException e) { Thread.currentThread().interrupt(); error = e; throw Exceptions.propagate(error); } } @Override public T next() { if (error != null) { // If any error has already been thrown, throw it again. throw Exceptions.propagate(error); } if (hasNext()) { isNextConsumed = true; return next; } else { throw new NoSuchElementException("No more elements"); } } @Override public void remove() { throw new UnsupportedOperationException("Read only iterator"); } } private static class NextObserver extends Subscriber> { private final BlockingQueue> buf = new ArrayBlockingQueue>(1); volatile int waiting; @SuppressWarnings("rawtypes") static final AtomicIntegerFieldUpdater WAITING_UPDATER = AtomicIntegerFieldUpdater.newUpdater(NextObserver.class, "waiting"); @Override public void onCompleted() { // ignore } @Override public void onError(Throwable e) { // ignore } @Override public void onNext(Notification args) { if (WAITING_UPDATER.getAndSet(this, 0) == 1 || !args.isOnNext()) { Notification toOffer = args; while (!buf.offer(toOffer)) { Notification concurrentItem = buf.poll(); // in case if we won race condition with onComplete/onError method if (concurrentItem != null && !concurrentItem.isOnNext()) { toOffer = concurrentItem; } } } } public Notification takeNext() throws InterruptedException { setWaiting(1); return buf.take(); } void setWaiting(int value) { waiting = value; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy