Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
com.landawn.abacus.util.Observer Maven / Gradle / Ivy
/*
* Copyright (C) 2017 HaiYang Li
*
* 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 com.landawn.abacus.util;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashSet;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Queue;
import java.util.Set;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.Executor;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.ScheduledThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicLong;
import com.landawn.abacus.annotation.NonNull;
import com.landawn.abacus.util.function.Consumer;
import com.landawn.abacus.util.function.Function;
import com.landawn.abacus.util.function.Predicate;
/**
*
*
* @since 0.9
*
* @author Haiyang Li
*
*/
public abstract class Observer {
private static final Object COMPLETE_FLAG = new Object();
protected static final double INTERVAL_FACTOR = 3;
protected static final Runnable EMPTY_ACTION = new Runnable() {
@Override
public void run() {
// Do nothing;
}
};
protected static final Consumer ON_ERROR_MISSING = new Consumer() {
@Override
public void accept(Exception t) {
throw new RuntimeException(t);
}
};
protected static final Executor asyncExecutor = Executors.newFixedThreadPool(IOUtil.IS_PLATFORM_ANDROID ? IOUtil.CPU_CORES : 32);
protected static final ScheduledThreadPoolExecutor scheduler = new ScheduledThreadPoolExecutor(IOUtil.IS_PLATFORM_ANDROID ? IOUtil.CPU_CORES : 32);
static {
scheduler.setRemoveOnCancelPolicy(true);
}
protected final Map, Long> scheduledFutures = new LinkedHashMap<>();
protected final Dispatcher dispatcher;
protected boolean hasMore = true;
protected Observer() {
this(new Dispatcher<>());
}
protected Observer(Dispatcher dispatcher) {
this.dispatcher = dispatcher;
}
@SuppressWarnings("rawtypes")
public static void complete(BlockingQueue> queue) {
((Queue) queue).offer(COMPLETE_FLAG);
}
public static Observer of(final BlockingQueue queue) {
N.checkArgNotNull(queue, "queue");
return new BlockingQueueObserver<>(queue);
}
public static Observer of(final Collection c) {
return of(N.isNullOrEmpty(c) ? ObjIterator. empty() : c.iterator());
}
public static Observer of(final Iterator iter) {
N.checkArgNotNull(iter, "iterator");
return new IteratorObserver<>(iter);
}
/**
*
* @param delayInMillis
* @return
* @see RxJava#timer
*/
public static Observer timer(long delayInMillis) {
return timer(delayInMillis, TimeUnit.MILLISECONDS);
}
/**
*
* @param delay
* @param unit
* @return
* @see RxJava#timer
*/
public static Observer timer(long delay, TimeUnit unit) {
N.checkArgument(delay >= 0, "delay can't be negative");
N.checkArgNotNull(unit, "Time unit can't be null");
return new TimerObserver<>(delay, unit);
}
/**
*
* @param periodInMillis
* @return
* @see RxJava#interval
*/
public static Observer interval(long periodInMillis) {
return interval(0, periodInMillis, TimeUnit.MILLISECONDS);
}
/**
* @param initialDelayInMillis
* @param periodInMillis
* @return
* @see RxJava#interval
*/
public static Observer interval(long initialDelayInMillis, long periodInMillis) {
return interval(initialDelayInMillis, periodInMillis, TimeUnit.MILLISECONDS);
}
/**
*
* @param period
* @param unit
* @return
* @see RxJava#interval
*/
public static Observer interval(long period, TimeUnit unit) {
return interval(0, period, unit);
}
/**
*
* @param initialDelay
* @param period
* @param unit
* @return
* @see RxJava#interval
*/
public static Observer interval(long initialDelay, long period, TimeUnit unit) {
N.checkArgument(initialDelay >= 0, "initialDelay can't be negative");
N.checkArgument(period > 0, "period can't be 0 or negative");
N.checkArgNotNull(unit, "Time unit can't be null");
return new IntervalObserver<>(initialDelay, period, unit);
}
/**
*
* @param intervalDurationInMillis
* @return this instance.
* @see RxJava#debounce
*/
public Observer debounce(final long intervalDurationInMillis) {
return debounce(intervalDurationInMillis, TimeUnit.MILLISECONDS);
}
/**
*
* @param intervalDuration
* @param unit
* @return this instance.
* @see RxJava#debounce
*/
public Observer debounce(final long intervalDuration, final TimeUnit unit) {
N.checkArgument(intervalDuration >= 0, "Interval can't be negative");
N.checkArgNotNull(unit, "Time unit can't be null");
if (intervalDuration == 0) {
return this;
}
final long intervalDurationInMillis = unit.toMillis(intervalDuration);
dispatcher.append(new Dispatcher() {
private long prevTimestamp = 0;
private long lastScheduledTime = 0;
@Override
public void onNext(final Object param) {
synchronized (holder) {
final long now = DateUtil.currentMillis();
if (holder.value() == N.NULL_MASK || now - lastScheduledTime > intervalDurationInMillis * INTERVAL_FACTOR) {
holder.setValue(param);
prevTimestamp = now;
schedule(intervalDuration, unit);
} else {
holder.setValue(param);
prevTimestamp = now;
}
}
}
private void schedule(final long delay, final TimeUnit unit) {
try {
scheduler.schedule(new Runnable() {
@Override
public void run() {
final long pastIntervalInMills = DateUtil.currentMillis() - prevTimestamp;
if (pastIntervalInMills >= intervalDurationInMillis) {
Object lastParam = null;
synchronized (holder) {
lastParam = holder.value();
holder.setValue(N.NULL_MASK);
}
if (lastParam != N.NULL_MASK && downDispatcher != null) {
downDispatcher.onNext(lastParam);
}
} else {
schedule(intervalDurationInMillis - pastIntervalInMills, TimeUnit.MILLISECONDS);
}
}
}, delay, unit);
lastScheduledTime = DateUtil.currentMillis();
} catch (Exception e) {
holder.setValue(N.NULL_MASK);
if (downDispatcher != null) {
downDispatcher.onError(e);
}
}
}
});
return this;
}
/**
*
* @param intervalDurationInMillis
* @return this instance.
* @see RxJava#throttleFirst
*/
public Observer throttleFirst(final long intervalDurationInMillis) {
return throttleFirst(intervalDurationInMillis, TimeUnit.MILLISECONDS);
}
/**
*
* @param intervalDuration
* @param unit
* @return this instance.
* @see RxJava#throttleFirst
*/
public Observer throttleFirst(final long intervalDuration, final TimeUnit unit) {
N.checkArgument(intervalDuration >= 0, "Interval can't be negative");
N.checkArgNotNull(unit, "Time unit can't be null");
if (intervalDuration == 0) {
return this;
}
final long intervalDurationInMillis = unit.toMillis(intervalDuration);
dispatcher.append(new Dispatcher() {
private long lastScheduledTime = 0;
@Override
public void onNext(final Object param) {
synchronized (holder) {
final long now = DateUtil.currentMillis();
if (holder.value() == N.NULL_MASK || now - lastScheduledTime > intervalDurationInMillis * INTERVAL_FACTOR) {
holder.setValue(param);
try {
scheduler.schedule(new Runnable() {
@Override
public void run() {
Object firstParam = null;
synchronized (holder) {
firstParam = holder.value();
holder.setValue(N.NULL_MASK);
}
if (firstParam != N.NULL_MASK && downDispatcher != null) {
downDispatcher.onNext(firstParam);
}
}
}, intervalDuration, unit);
lastScheduledTime = now;
} catch (Exception e) {
holder.setValue(N.NULL_MASK);
if (downDispatcher != null) {
downDispatcher.onError(e);
}
}
}
}
}
});
return this;
}
/**
*
* @param intervalDurationInMillis
* @return this instance.
* @see RxJava#throttleLast
*/
public Observer throttleLast(final long intervalDurationInMillis) {
return throttleLast(intervalDurationInMillis, TimeUnit.MILLISECONDS);
}
/**
*
* @param intervalDuration
* @param unit
* @return this instance.
* @see RxJava#throttleLast
*/
public Observer throttleLast(final long intervalDuration, final TimeUnit unit) {
N.checkArgument(intervalDuration >= 0, "Delay can't be negative");
N.checkArgNotNull(unit, "Time unit can't be null");
if (intervalDuration == 0) {
return this;
}
final long intervalDurationInMillis = unit.toMillis(intervalDuration);
dispatcher.append(new Dispatcher() {
private long lastScheduledTime = 0;
@Override
public void onNext(final Object param) {
synchronized (holder) {
final long now = DateUtil.currentMillis();
if (holder.value() == N.NULL_MASK || now - lastScheduledTime > intervalDurationInMillis * INTERVAL_FACTOR) {
holder.setValue(param);
try {
scheduler.schedule(new Runnable() {
@Override
public void run() {
Object lastParam = null;
synchronized (holder) {
lastParam = holder.value();
holder.setValue(N.NULL_MASK);
}
if (lastParam != N.NULL_MASK && downDispatcher != null) {
downDispatcher.onNext(lastParam);
}
}
}, intervalDuration, unit);
lastScheduledTime = now;
} catch (Exception e) {
holder.setValue(N.NULL_MASK);
if (downDispatcher != null) {
downDispatcher.onError(e);
}
}
} else {
holder.setValue(param);
}
}
}
});
return this;
}
/**
*
* @param delayInMillis
* @return this instance.
* @see RxJava#delay
*/
public Observer delay(final long delayInMillis) {
return delay(delayInMillis, TimeUnit.MILLISECONDS);
}
/**
*
* @param delayInMillis
* @param unit
* @return this instance.
* @see RxJava#delay
*/
public Observer delay(final long delay, final TimeUnit unit) {
N.checkArgument(delay >= 0, "Delay can't be negative");
N.checkArgNotNull(unit, "Time unit can't be null");
if (delay == 0) {
return this;
}
dispatcher.append(new Dispatcher() {
private final long startTime = DateUtil.currentMillis();
private boolean isDelayed = false;
@Override
public void onNext(final Object param) {
if (isDelayed == false) {
N.sleep(unit.toMillis(delay) - (DateUtil.currentMillis() - startTime));
isDelayed = true;
}
if (downDispatcher != null) {
downDispatcher.onNext(param);
}
}
});
return this;
}
/**
*
* @return this instance.
* @see RxJava#timeInterval
*/
public Observer> timeInterval() {
dispatcher.append(new Dispatcher() {
private long startTime = DateUtil.currentMillis();
@Override
public synchronized void onNext(final Object param) {
if (downDispatcher != null) {
long now = DateUtil.currentMillis();
long interval = now - startTime;
startTime = now;
downDispatcher.onNext(Timed.of(param, interval));
}
}
});
return (Observer>) this;
}
/**
*
* @return this instance.
* @see RxJava#timestamp
*/
public Observer> timestamp() {
dispatcher.append(new Dispatcher() {
@Override
public void onNext(final Object param) {
if (downDispatcher != null) {
downDispatcher.onNext(Timed.of(param, DateUtil.currentMillis()));
}
}
});
return (Observer>) this;
}
public Observer skip(final long n) {
N.checkArgument(n >= 0, "n can't be negative");
if (n > 0) {
dispatcher.append(new Dispatcher() {
private final AtomicLong counter = new AtomicLong();
@Override
public void onNext(final Object param) {
if (downDispatcher != null && counter.incrementAndGet() > n) {
downDispatcher.onNext(param);
}
}
});
}
return this;
}
public Observer limit(final long n) {
N.checkArgument(n >= 0, "n can't be negative");
dispatcher.append(new Dispatcher() {
private final AtomicLong counter = new AtomicLong();
@Override
public void onNext(final Object param) {
if (downDispatcher != null && counter.incrementAndGet() <= n) {
downDispatcher.onNext(param);
} else {
hasMore = false;
}
}
});
return this;
}
/**
*
* @return
*/
public Observer distinct() {
dispatcher.append(new Dispatcher() {
private Set set = new HashSet<>();
@Override
public void onNext(final Object param) {
if (downDispatcher != null && set.add((T) param)) {
downDispatcher.onNext(param);
}
}
});
return this;
}
/**
*
* @param keyExtractor
* @return
*/
public Observer distinctBy(final Function super T, ?> keyExtractor) {
dispatcher.append(new Dispatcher() {
private Set set = new HashSet<>();
@Override
public void onNext(final Object param) {
if (downDispatcher != null && set.add(keyExtractor.apply((T) param))) { // onError if keyExtractor.apply throws exception?
downDispatcher.onNext(param);
}
}
});
return this;
}
public Observer filter(final Predicate super T> filter) {
dispatcher.append(new Dispatcher() {
@Override
public void onNext(final Object param) {
if (downDispatcher != null && filter.test((T) param) == true) { // onError if filter.test throws exception?
downDispatcher.onNext(param);
}
}
});
return this;
}
public Observer map(final Function super T, U> map) {
dispatcher.append(new Dispatcher() {
@Override
public void onNext(final Object param) {
if (downDispatcher != null) {
downDispatcher.onNext(map.apply((T) param)); // onError if map.apply throws exception?
}
}
});
return (Observer) this;
}
public Observer flatMap(final Function super T, Collection> map) {
dispatcher.append(new Dispatcher() {
@Override
public void onNext(final Object param) {
if (downDispatcher != null) {
final Collection c = map.apply((T) param); // onError if map.apply throws exception?
if (N.notNullOrEmpty(c)) {
for (U u : c) {
downDispatcher.onNext(u);
}
}
}
}
});
return (Observer) this;
}
/**
*
* @param timespan
* @param unit
* @return this instance
* @see RxJava#window(long, java.util.concurrent.TimeUnit)
*/
public Observer> buffer(final long timespan, final TimeUnit unit) {
return buffer(timespan, unit, Integer.MAX_VALUE);
}
/**
*
* @param timespan
* @param unit
* @return this instance
* @see RxJava#window(long, java.util.concurrent.TimeUnit, int)
*/
public Observer> buffer(final long timespan, final TimeUnit unit, final int count) {
N.checkArgument(timespan > 0, "timespan can't be 0 or negative");
N.checkArgNotNull(unit, "Time unit can't be null");
N.checkArgument(count > 0, "count can't be 0 or negative");
dispatcher.append(new Dispatcher() {
private final List queue = new ArrayList<>();
{
scheduledFutures.put(scheduler.scheduleAtFixedRate(new Runnable() {
@Override
public void run() {
List list = null;
synchronized (queue) {
list = new ArrayList<>(queue);
queue.clear();
}
if (downDispatcher != null) {
downDispatcher.onNext(list);
}
}
}, timespan, timespan, unit), timespan);
}
@Override
public void onNext(final Object param) {
List list = null;
synchronized (queue) {
queue.add((T) param);
if (queue.size() == count) {
list = new ArrayList<>(queue);
queue.clear();
}
}
if (list != null && downDispatcher != null) {
downDispatcher.onNext(list);
}
}
});
return (Observer>) this;
}
/**
*
* @param timespan
* @param timeskip
* @param unit
* @return
* @see RxJava#window(long, long, java.util.concurrent.TimeUnit)
*/
public Observer> buffer(final long timespan, final long timeskip, final TimeUnit unit) {
return buffer(timespan, timeskip, unit, Integer.MAX_VALUE);
}
/**
*
* @param timespan
* @param timeskip
* @param unit
* @param count
* @return
* @see RxJava#window(long, long, java.util.concurrent.TimeUnit)
*/
public Observer> buffer(final long timespan, final long timeskip, final TimeUnit unit, final int count) {
N.checkArgument(timespan > 0, "timespan can't be 0 or negative");
N.checkArgument(timeskip > 0, "timeskip can't be 0 or negative");
N.checkArgNotNull(unit, "Time unit can't be null");
N.checkArgument(count > 0, "count can't be 0 or negative");
dispatcher.append(new Dispatcher() {
private final long startTime = DateUtil.currentMillis();
private final long interval = timespan + timeskip;
private final List queue = new ArrayList<>();
{
scheduledFutures.put(scheduler.scheduleAtFixedRate(new Runnable() {
@Override
public void run() {
List list = null;
synchronized (queue) {
list = new ArrayList<>(queue);
queue.clear();
}
if (downDispatcher != null) {
downDispatcher.onNext(list);
}
}
}, timespan, interval, unit), interval);
}
@Override
public void onNext(final Object param) {
if ((DateUtil.currentMillis() - startTime) % interval <= timespan) {
List list = null;
synchronized (queue) {
queue.add((T) param);
if (queue.size() == count) {
list = new ArrayList<>(queue);
queue.clear();
}
}
if (list != null && downDispatcher != null) {
downDispatcher.onNext(list);
}
}
}
});
return (Observer>) this;
}
public void observe(final Consumer super T> action) {
observe(action, ON_ERROR_MISSING);
}
public void observe(final Consumer super T> action, final Consumer super Exception> onError) {
observe(action, onError, EMPTY_ACTION);
}
public abstract void observe(final Consumer super T> action, final Consumer super Exception> onError, final Runnable onComplete);
void cancelScheduledFutures() {
final long startTime = DateUtil.currentMillis();
if (N.notNullOrEmpty(scheduledFutures)) {
for (Map.Entry, Long> entry : scheduledFutures.entrySet()) {
final long delay = entry.getValue();
N.sleep(delay - (DateUtil.currentMillis() - startTime)
+ delay /* Extending another delay just want to make sure last schedule can be completed before the schedule task is cancelled*/);
entry.getKey().cancel(false);
}
}
}
protected static class Node {
public final T value;
public Node next;
public Node(final T value) {
this(value, null);
}
public Node(final T value, Node next) {
this.value = value;
this.next = next;
}
}
protected static class Dispatcher {
protected final Holder holder = Holder.of(N.NULL_MASK);
protected Dispatcher downDispatcher;
public void onNext(@NonNull final T value) {
if (downDispatcher != null) {
downDispatcher.onNext(value);
}
}
/**
* Signal a Exception exception.
* @param error the Exception to signal, not null
*/
public void onError(@NonNull final Exception error) {
if (downDispatcher != null) {
downDispatcher.onError(error);
}
}
/**
* Signal a completion.
*/
public void onComplete() {
if (downDispatcher != null) {
downDispatcher.onComplete();
}
}
public void append(Dispatcher downDispatcher) {
Dispatcher tmp = this;
while (tmp.downDispatcher != null) {
tmp = tmp.downDispatcher;
}
tmp.downDispatcher = downDispatcher;
}
}
protected static abstract class DispatcherBase extends Dispatcher {
private final Consumer super Exception> onError;
private final Runnable onComplete;
protected DispatcherBase(final Consumer super Exception> onError, final Runnable onComplete) {
this.onError = onError;
this.onComplete = onComplete;
}
@Override
public void onError(final Exception error) {
onError.accept(error);
}
@Override
public void onComplete() {
onComplete.run();
}
}
protected static abstract class ObserverBase extends Observer {
protected ObserverBase() {
}
}
static final class BlockingQueueObserver extends ObserverBase {
private final BlockingQueue queue;
BlockingQueueObserver(final BlockingQueue queue) {
this.queue = queue;
}
@Override
public void observe(final Consumer super T> action, final Consumer super Exception> onError, final Runnable onComplete) {
N.checkArgNotNull(action, "action");
dispatcher.append(new DispatcherBase(onError, onComplete) {
@Override
public void onNext(Object param) {
action.accept((T) param);
}
});
asyncExecutor.execute(new Runnable() {
@Override
public void run() {
T next = null;
boolean isOnError = true;
try {
while (hasMore && (next = queue.poll(Long.MAX_VALUE, TimeUnit.MILLISECONDS)) != COMPLETE_FLAG) {
isOnError = false;
dispatcher.onNext(next);
isOnError = true;
}
isOnError = false;
onComplete.run();
} catch (Exception e) {
if (isOnError) {
onError.accept(e);
} else {
throw N.toRuntimeException(e);
}
} finally {
cancelScheduledFutures();
}
}
});
}
}
static final class IteratorObserver extends ObserverBase {
private final Iterator iter;
IteratorObserver(final Iterator iter) {
this.iter = iter;
}
@Override
public void observe(final Consumer super T> action, final Consumer super Exception> onError, final Runnable onComplete) {
N.checkArgNotNull(action, "action");
dispatcher.append(new DispatcherBase(onError, onComplete) {
@Override
public void onNext(Object param) {
action.accept((T) param);
}
});
asyncExecutor.execute(new Runnable() {
@Override
public void run() {
boolean isOnError = true;
try {
while (hasMore && iter.hasNext()) {
isOnError = false;
dispatcher.onNext(iter.next());
isOnError = true;
}
isOnError = false;
onComplete.run();
} catch (Exception e) {
if (isOnError) {
onError.accept(e);
} else {
throw N.toRuntimeException(e);
}
} finally {
cancelScheduledFutures();
}
}
});
}
}
static final class TimerObserver extends ObserverBase {
private final long delay;
private final TimeUnit unit;
TimerObserver(long delay, TimeUnit unit) {
this.delay = delay;
this.unit = unit;
}
@Override
public void observe(final Consumer super T> action, final Consumer super Exception> onError, final Runnable onComplete) {
N.checkArgNotNull(action, "action");
dispatcher.append(new DispatcherBase(onError, onComplete) {
@Override
public void onNext(Object param) {
action.accept((T) param);
}
});
scheduler.schedule(new Runnable() {
@Override
public void run() {
try {
dispatcher.onNext(0L);
onComplete.run();
} finally {
cancelScheduledFutures();
}
}
}, delay, unit);
}
}
static final class IntervalObserver extends ObserverBase {
private final long initialDelay;
private final long period;
private final TimeUnit unit;
private ScheduledFuture> future = null;
IntervalObserver(long initialDelay, long period, TimeUnit unit) {
this.initialDelay = initialDelay;
this.period = period;
this.unit = unit;
}
@Override
public void observe(final Consumer super T> action, final Consumer super Exception> onError, final Runnable onComplete) {
N.checkArgNotNull(action, "action");
dispatcher.append(new DispatcherBase(onError, onComplete) {
@Override
public void onNext(Object param) {
action.accept((T) param);
}
});
future = scheduler.scheduleAtFixedRate(new Runnable() {
private long val = 0;
@Override
public void run() {
if (hasMore == false) {
try {
dispatcher.onComplete();
} finally {
try {
future.cancel(true);
} finally {
cancelScheduledFutures();
}
}
} else {
try {
dispatcher.onNext(val++);
} catch (Exception e) {
try {
future.cancel(true);
} finally {
cancelScheduledFutures();
}
}
}
}
}, initialDelay, period, unit);
}
}
}