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

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

There is a newer version: 1.3.8
Show newest version
/**
 * 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.ArrayList;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;

import rx.Observable;
import rx.Observable.Operator;
import rx.Producer;
import rx.Subscriber;
import rx.exceptions.Exceptions;

/**
 * This operation takes
 * values from the specified {@link Observable} source and stores them in all active chunks until the buffer
 * contains a specified number of elements. The buffer is then emitted. Chunks are created after a certain
 * amount of values have been received. When the source {@link Observable} completes or produces an error,
 * the currently active chunks are emitted, and the event is propagated to all subscribed {@link Subscriber}s.
 * 

* Note that this operation can produce non-connected, connected non-overlapping, or overlapping * chunks depending on the input parameters. *

* @param the buffered value type */ public final class OperatorBufferWithSize implements Operator, T> { final int count; final int skip; /** * @param count * the number of elements a buffer should have before being emitted * @param skip * the interval with which chunks have to be created. Note that when {@code skip == count} * the operator will produce non-overlapping chunks. If * {@code skip < count}, this buffer operation will produce overlapping chunks and if * {@code skip > count} non-overlapping chunks will be created and some values will not be pushed * into a buffer at all! */ public OperatorBufferWithSize(int count, int skip) { if (count <= 0) { throw new IllegalArgumentException("count must be greater than 0"); } if (skip <= 0) { throw new IllegalArgumentException("skip must be greater than 0"); } this.count = count; this.skip = skip; } @Override public Subscriber call(final Subscriber> child) { if (count == skip) { return new Subscriber(child) { List buffer; @Override public void setProducer(final Producer producer) { child.setProducer(new Producer() { private volatile boolean infinite = false; @Override public void request(long n) { if (infinite) { return; } if (n >= Long.MAX_VALUE / count) { // n == Long.MAX_VALUE or n * count >= Long.MAX_VALUE infinite = true; producer.request(Long.MAX_VALUE); } else { producer.request(n * count); } } }); } @Override public void onNext(T t) { if (buffer == null) { buffer = new ArrayList(count); } buffer.add(t); if (buffer.size() == count) { List oldBuffer = buffer; buffer = null; child.onNext(oldBuffer); } } @Override public void onError(Throwable e) { buffer = null; child.onError(e); } @Override public void onCompleted() { List oldBuffer = buffer; buffer = null; if (oldBuffer != null) { try { child.onNext(oldBuffer); } catch (Throwable t) { Exceptions.throwOrReport(t, this); return; } } child.onCompleted(); } }; } return new Subscriber(child) { final List> chunks = new LinkedList>(); int index; @Override public void setProducer(final Producer producer) { child.setProducer(new Producer() { private volatile boolean firstRequest = true; private volatile boolean infinite = false; private void requestInfinite() { infinite = true; producer.request(Long.MAX_VALUE); } @Override public void request(long n) { if (n == 0) { return; } if (n < 0) { throw new IllegalArgumentException("request a negative number: " + n); } if (infinite) { return; } if (n == Long.MAX_VALUE) { requestInfinite(); } else { if (firstRequest) { firstRequest = false; if (n - 1 >= (Long.MAX_VALUE - count) / skip) { // count + skip * (n - 1) >= Long.MAX_VALUE requestInfinite(); return; } // count = 5, skip = 2, n = 3 // * * * * * // * * * * * // * * * * * // request = 5 + 2 * ( 3 - 1) producer.request(count + skip * (n - 1)); } else { if (n >= Long.MAX_VALUE / skip) { // skip * n >= Long.MAX_VALUE requestInfinite(); return; } // count = 5, skip = 2, n = 3 // (* * *) * * // ( *) * * * * // * * * * * // request = skip * n // "()" means the items already emitted before this request producer.request(skip * n); } } } }); } @Override public void onNext(T t) { if (index++ % skip == 0) { chunks.add(new ArrayList(count)); } Iterator> it = chunks.iterator(); while (it.hasNext()) { List chunk = it.next(); chunk.add(t); if (chunk.size() == count) { it.remove(); child.onNext(chunk); } } } @Override public void onError(Throwable e) { chunks.clear(); child.onError(e); } @Override public void onCompleted() { try { for (List chunk : chunks) { try { child.onNext(chunk); } catch (Throwable t) { Exceptions.throwOrReport(t, this); return; } } child.onCompleted(); } finally { chunks.clear(); } } }; } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy