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

hu.akarnokd.rxjava2.string.FlowableCharSequence Maven / Gradle / Ivy

There is a newer version: 0.20.10
Show newest version
/*
 * Copyright 2016 David Karnok
 *
 * 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 hu.akarnokd.rxjava2.string;

import java.util.concurrent.atomic.AtomicLongFieldUpdater;

import org.reactivestreams.Subscriber;

import io.reactivex.Flowable;
import io.reactivex.internal.fuseable.QueueSubscription;
import io.reactivex.internal.subscriptions.*;
import io.reactivex.internal.util.BackpressureHelper;

/**
 * Streams the characters of a string.
 */
final class FlowableCharSequence extends Flowable {

    final CharSequence string;

    FlowableCharSequence(CharSequence string) {
        this.string = string;
    }

    @Override
    public void subscribeActual(Subscriber s) {
        s.onSubscribe(new CharSequenceSubscription(s, string));
    }

    static final class CharSequenceSubscription
    extends BasicQueueSubscription {

        private static final long serialVersionUID = -4593793201463047197L;

        final Subscriber actual;

        final CharSequence string;

        final int end;

        int index;

        volatile boolean cancelled;

        volatile long requested;
        static final AtomicLongFieldUpdater REQUESTED =
                AtomicLongFieldUpdater.newUpdater(CharSequenceSubscription.class, "requested");

        CharSequenceSubscription(Subscriber actual, CharSequence string) {
            this.actual = actual;
            this.string = string;
            this.end = string.length();
        }

        @Override
        public void cancel() {
            cancelled = true;
        }

        @Override
        public void request(long n) {
            if (SubscriptionHelper.validate(n)) {
                if (BackpressureHelper.add(this, n) == 0) {
                    if (n == Long.MAX_VALUE) {
                        fastPath();
                    } else {
                        slowPath(n);
                    }
                }
            }
        }

        void fastPath() {
            int e = end;
            CharSequence s = string;
            Subscriber a = actual;

            for (int i = index; i != e; i++) {
                if (cancelled) {
                    return;
                }

                a.onNext((int)s.charAt(i));
            }

            if (!cancelled) {
                a.onComplete();
            }
        }

        void slowPath(long r) {
            long e = 0L;
            int i = index;
            int f = end;
            CharSequence s = string;
            Subscriber a = actual;

            for (;;) {

                while (e != r && i != f) {
                    if (cancelled) {
                        return;
                    }

                    a.onNext((int)s.charAt(i));

                    i++;
                    e++;
                }

                if (i == f) {
                    if (!cancelled) {
                        a.onComplete();
                    }
                    return;
                }

                r = requested;
                if (e == r) {
                    index = i;
                    r = REQUESTED.addAndGet(this, -e);
                    if (r == 0L) {
                        break;
                    }
                }
            }
        }

        @Override
        public int requestFusion(int requestedMode) {
            return requestedMode & QueueSubscription.SYNC;
        }

        @Override
        public Integer poll() {
            int i = index;
            if (i != end) {
                index = i + 1;
                return (int)string.charAt(i);
            }
            return null;
        }

        @Override
        public boolean isEmpty() {
            return index != end;
        }

        @Override
        public void clear() {
            index = end;
        }
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy