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

rx.internal.operators.NotificationLite 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.io.Serializable;

import rx.Notification.Kind;
import rx.Observer;

/**
 * For use in internal operators that need something like materialize and dematerialize wholly within the
 * implementation of the operator but don't want to incur the allocation cost of actually creating
 * {@link rx.Notification} objects for every {@link Observer#onNext onNext} and
 * {@link Observer#onCompleted onCompleted}.
 * 

* An object is allocated inside {@link #error(Throwable)} to wrap the {@link Throwable} but this shouldn't * affect performance because exceptions should be exceptionally rare. *

* It's implemented as a singleton to maintain some semblance of type safety that is completely non-existent. * * @param the element type */ public final class NotificationLite { @SuppressWarnings("rawtypes") private static final NotificationLite INSTANCE = new NotificationLite(); private NotificationLite() { // singleton } /** * Gets the {@code NotificationLite} singleton. * * @param the value type * @return the sole {@code NotificationLite} object */ @SuppressWarnings("unchecked") public static NotificationLite instance() { return INSTANCE; } private static final Object ON_COMPLETED_SENTINEL = new Serializable() { private static final long serialVersionUID = 1; @Override public String toString() { return "Notification=>Completed"; } }; private static final Object ON_NEXT_NULL_SENTINEL = new Serializable() { private static final long serialVersionUID = 2; @Override public String toString() { return "Notification=>NULL"; } }; static final class OnErrorSentinel implements Serializable { private static final long serialVersionUID = 3; final Throwable e; public OnErrorSentinel(Throwable e) { this.e = e; } @Override public String toString() { return "Notification=>Error:" + e; } } /** * Creates a lite {@code onNext} notification for the value passed in without doing any allocation. Can * be unwrapped and sent with the {@link #accept} method. * * @param t * the item emitted to {@code onNext} * @return the item, or a null token representing the item if the item is {@code null} */ public Object next(T t) { if (t == null) { return ON_NEXT_NULL_SENTINEL; } else { return t; } } /** * Creates a lite {@code onCompleted} notification without doing any allocation. Can be unwrapped and * sent with the {@link #accept} method. * * @return a completion token */ public Object completed() { return ON_COMPLETED_SENTINEL; } /** * Create a lite {@code onError} notification. This call creates an object to wrap the {@link Throwable}, * but since there should only be one of these, the performance impact should be small. Can be unwrapped and * sent with the {@link #accept} method. * * @param e * the {@code Throwable} in the {@code onError} notification * @return an object encapsulating the exception */ public Object error(Throwable e) { return new OnErrorSentinel(e); } /** * Unwraps the lite notification and calls the appropriate method on the {@link Observer}. * * @param o * the {@link Observer} to call {@code onNext}, {@code onCompleted}, or {@code onError}. * @param n * the lite notification * @return {@code true} if {@code n} represents a termination event; {@code false} otherwise * @throws IllegalArgumentException * if the notification is null. * @throws NullPointerException * if the {@link Observer} is null. */ @SuppressWarnings("unchecked") public boolean accept(Observer o, Object n) { if (n == ON_COMPLETED_SENTINEL) { o.onCompleted(); return true; } else if (n == ON_NEXT_NULL_SENTINEL) { o.onNext(null); return false; } else if (n != null) { if (n.getClass() == OnErrorSentinel.class) { o.onError(((OnErrorSentinel) n).e); return true; } o.onNext((T) n); return false; } else { throw new IllegalArgumentException("The lite notification can not be null"); } } /** * Indicates whether or not the lite notification represents an {@code onCompleted} event. * * @param n * the lite notification * @return {@code true} if {@code n} represents an {@code onCompleted} event; {@code false} otherwise */ public boolean isCompleted(Object n) { return n == ON_COMPLETED_SENTINEL; } /** * Indicates whether or not the lite notification represents an {@code onError} event. * * @param n * the lite notification * @return {@code true} if {@code n} represents an {@code onError} event; {@code false} otherwise */ public boolean isError(Object n) { return n instanceof OnErrorSentinel; } /** * Indicates whether or not the lite notification represents a wrapped {@code null} {@code onNext} event. * @param n the lite notification * @return {@code true} if {@code n} represents a wrapped {@code null} {@code onNext} event, {@code false} otherwise */ public boolean isNull(Object n) { return n == ON_NEXT_NULL_SENTINEL; } /** * Indicates whether or not the lite notification represents an {@code onNext} event. * @param n the lite notification * @return {@code true} if {@code n} represents an {@code onNext} event, {@code false} otherwise */ public boolean isNext(Object n) { return n != null && !isError(n) && !isCompleted(n); } /** * Indicates which variety a particular lite notification is. If you need something more complex than * simply calling the right method on an {@link Observer} then you can use this method to get the * {@link rx.Notification.Kind}. * * @param n * the lite notification * @throws IllegalArgumentException * if the notification is null. * @return the {@link Kind} of lite notification {@code n} is: either {@code Kind.OnCompleted}, * {@code Kind.OnError}, or {@code Kind.OnNext} */ public Kind kind(Object n) { if (n == null) { throw new IllegalArgumentException("The lite notification can not be null"); } else if (n == ON_COMPLETED_SENTINEL) { return Kind.OnCompleted; } else if (n instanceof OnErrorSentinel) { return Kind.OnError; } else { // value or ON_NEXT_NULL_SENTINEL but either way it's an OnNext return Kind.OnNext; } } /** * Returns the item corresponding to this {@code OnNext} lite notification. Bad things happen if you pass * this an {@code OnComplete} or {@code OnError} notification type. For performance reasons, this method * does not check for this, so you are expected to prevent such a mishap. * * @param n * the lite notification (of type {@code Kind.OnNext}) * @return the unwrapped value, which can be null */ @SuppressWarnings("unchecked") public T getValue(Object n) { return n == ON_NEXT_NULL_SENTINEL ? null : (T) n; } /** * Returns the {@link Throwable} corresponding to this {@code OnError} lite notification. Bad things happen * if you pass this an {@code OnComplete} or {@code OnNext} notification type. For performance reasons, this * method does not check for this, so you are expected to prevent such a mishap. * * @param n * the lite notification (of type {@code Kind.OnError}) * @return the {@link Throwable} wrapped inside {@code n} */ public Throwable getError(Object n) { return ((OnErrorSentinel) n).e; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy