com.landawn.abacus.android.util.Async Maven / Gradle / Ivy
/*
* Copyright (c) 2015, 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.android.util;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Executor;
import java.util.concurrent.Future;
import java.util.concurrent.FutureTask;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.ScheduledThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import com.landawn.abacus.util.DateUtil;
import com.landawn.abacus.util.MoreExecutors;
import com.landawn.abacus.util.Retry;
import com.landawn.abacus.util.Try;
import com.landawn.abacus.util.function.BiPredicate;
import com.landawn.abacus.util.function.Predicate;
import android.os.AsyncTask;
import android.os.Handler;
import android.os.Looper;
/**
*
* @since 0.8
*
* @author Haiyang Li
*/
public class Async {
static final ScheduledExecutorService SCHEDULED_EXECUTOR;
static {
final ScheduledThreadPoolExecutor executor = new ScheduledThreadPoolExecutor(8);
executor.setKeepAliveTime(180, TimeUnit.SECONDS);
executor.allowCoreThreadTimeOut(true);
executor.setRemoveOnCancelPolicy(true);
SCHEDULED_EXECUTOR = MoreExecutors.getExitingScheduledExecutorService(executor);
}
static final _UIExecutor _UI_EXECUTOR = new _UIExecutor();
public static final Executor SERIAL_EXECUTOR = AsyncTask.SERIAL_EXECUTOR;
public static final Executor TP_EXECUTOR = AsyncTask.THREAD_POOL_EXECUTOR;
public static final Executor UI_EXECUTOR = _UI_EXECUTOR;
private Async() {
// Singleton
}
/**
* The action will be asynchronously executed with {@code android.io.AsyncTask#SERIAL_EXECUTOR} in background.
*
* @param action
* @return
*/
static CompletableFuture execute(final Runnable action) {
return execute(new FutureTask(action, null), SERIAL_EXECUTOR);
}
static CompletableFuture execute(final Runnable action, final long delay) {
final Callable> scheduledAction = new Callable>() {
@Override
public CompletableFuture call() throws Exception {
return Async.execute(action);
}
};
final ScheduledFuture> scheduledFuture = SCHEDULED_EXECUTOR.schedule(scheduledAction, delay, TimeUnit.MILLISECONDS);
return new CompletableFuture<>(wrap(scheduledFuture), null, SERIAL_EXECUTOR);
}
/**
*
* @param action
* @param retryTimes
* @param retryInterval
* @param retryCondition
* @return
*/
static CompletableFuture execute(final Try.Runnable extends Exception> action, final int retryTimes, final long retryInterval,
final Predicate super Exception> retryCondition) {
return execute(new Callable() {
@Override
public Void call() throws Exception {
Retry.of(retryTimes, retryInterval, retryCondition).run(action);
return null;
}
});
}
/**
* The action will be asynchronously executed with {@code android.io.AsyncTask#SERIAL_EXECUTOR} in background.
*
* @param action
* @return
*/
static CompletableFuture execute(final Callable action) {
return execute(new FutureTask<>(action), SERIAL_EXECUTOR);
}
static CompletableFuture execute(final Callable action, final long delay) {
final Callable> scheduledAction = new Callable>() {
@Override
public CompletableFuture call() throws Exception {
return Async.execute(action);
}
};
final ScheduledFuture> scheduledFuture = SCHEDULED_EXECUTOR.schedule(scheduledAction, delay, TimeUnit.MILLISECONDS);
return new CompletableFuture<>(wrap(scheduledFuture), null, SERIAL_EXECUTOR);
}
/**
*
* @param action
* @param retryTimes
* @param retryInterval
* @param retryCondition
* @return
*/
static CompletableFuture execute(final Callable action, final int retryTimes, final long retryInterval,
final BiPredicate super T, ? super Exception> retryCondition) {
return execute(new Callable() {
@Override
public T call() throws Exception {
final Retry retry = Retry.of(retryTimes, retryInterval, retryCondition);
return retry.call(action);
}
});
}
/**
* The action will be asynchronously executed with {@code android.io.AsyncTask#THREAD_POOL_EXECUTOR} in background.
*
* @param action
* @return
*/
static CompletableFuture executeWithThreadPool(final Runnable action) {
return execute(new FutureTask(action, null), TP_EXECUTOR);
}
static CompletableFuture executeWithThreadPool(final Runnable action, final long delay) {
final Callable> scheduledAction = new Callable>() {
@Override
public CompletableFuture call() throws Exception {
return Async.executeWithThreadPool(action);
}
};
final ScheduledFuture> scheduledFuture = SCHEDULED_EXECUTOR.schedule(scheduledAction, delay, TimeUnit.MILLISECONDS);
return new CompletableFuture<>(wrap(scheduledFuture), null, TP_EXECUTOR);
}
/**
*
* @param action
* @param retryTimes
* @param retryInterval
* @param retryCondition
* @return
*/
static CompletableFuture executeWithThreadPool(final Try.Runnable extends Exception> action, final int retryTimes, final long retryInterval,
final Predicate super Exception> retryCondition) {
return execute(new Callable() {
@Override
public Void call() throws Exception {
Retry.of(retryTimes, retryInterval, retryCondition).run(action);
return null;
}
});
}
/**
* The action will be asynchronously executed with {@code android.io.AsyncTask#THREAD_POOL_EXECUTOR} in background.
*
* @param action
* @return
*/
static CompletableFuture executeWithThreadPool(final Callable action) {
return execute(new FutureTask<>(action), TP_EXECUTOR);
}
static CompletableFuture executeWithThreadPool(final Callable action, final long delay) {
final Callable> scheduledAction = new Callable>() {
@Override
public CompletableFuture call() throws Exception {
return Async.executeWithThreadPool(action);
}
};
final ScheduledFuture> scheduledFuture = SCHEDULED_EXECUTOR.schedule(scheduledAction, delay, TimeUnit.MILLISECONDS);
return new CompletableFuture<>(wrap(scheduledFuture), null, TP_EXECUTOR);
}
/**
*
* @param action
* @param retryTimes
* @param retryInterval
* @param retryCondition
* @return
*/
static CompletableFuture executeWithThreadPool(final Callable action, final int retryTimes, final long retryInterval,
final BiPredicate super T, ? super Exception> retryCondition) {
return executeWithThreadPool(new Callable() {
@Override
public T call() throws Exception {
final Retry retry = Retry.of(retryTimes, retryInterval, retryCondition);
return retry.call(action);
}
});
}
/**
* The action will be asynchronously executed in UI thread.
*
* @param action
* @return
*/
static CompletableFuture executeOnUiThread(final Runnable action) {
return executeOnUiThread(action, 0);
}
/**
* The action will be asynchronously executed in UI thread.
*
* @param action
* @param delay
* @return
*/
static CompletableFuture executeOnUiThread(final Runnable action, final long delay) {
return execute(new FutureTask(action, null), _UI_EXECUTOR, delay);
}
/**
*
* @param action
* @param retryTimes
* @param retryInterval
* @param retryCondition
* @return
*/
static CompletableFuture executeOnUiThread(final Try.Runnable extends Exception> action, final int retryTimes, final long retryInterval,
final Predicate super Exception> retryCondition) {
return execute(new Callable() {
@Override
public Void call() throws Exception {
Retry.of(retryTimes, retryInterval, retryCondition).run(action);
return null;
}
});
}
/**
* The action will be asynchronously executed in UI thread.
*
* @param action
* @return
*/
static CompletableFuture executeOnUiThread(final Callable action) {
return executeOnUiThread(action, 0);
}
/**
* The action will be asynchronously executed in UI thread.
*
* @param action
* @param delay
* @return
*/
static CompletableFuture executeOnUiThread(final Callable action, final long delay) {
return execute(new FutureTask<>(action), _UI_EXECUTOR, delay);
}
/**
*
* @param action
* @param retryTimes
* @param retryInterval
* @param retryCondition
* @return
*/
static CompletableFuture executeOnUiThread(final Callable action, final int retryTimes, final long retryInterval,
final BiPredicate super T, ? super Exception> retryCondition) {
return executeOnUiThread(new Callable() {
@Override
public T call() throws Exception {
final Retry retry = Retry.of(retryTimes, retryInterval, retryCondition);
return retry.call(action);
}
});
}
private static CompletableFuture execute(final FutureTask futureTask, final Executor executor) {
executor.execute(futureTask);
return new CompletableFuture<>(futureTask, null, executor);
}
private static CompletableFuture execute(final FutureTask futureTask, final _UIExecutor executor, final long delay) {
executor.execute(futureTask, delay);
return new CompletableFuture<>(futureTask, null, executor);
}
private static Future wrap(final ScheduledFuture> scheduledFuture) {
return new Future() {
@Override
public boolean cancel(boolean mayInterruptIfRunning) {
if (scheduledFuture.cancel(mayInterruptIfRunning) && scheduledFuture.isDone()) {
try {
final CompletableFuture resFuture = scheduledFuture.get();
return resFuture == null || resFuture.cancel(mayInterruptIfRunning);
} catch (Exception e) {
return false;
}
}
return false;
}
@Override
public boolean isCancelled() {
if (scheduledFuture.isCancelled() && scheduledFuture.isDone()) {
try {
final CompletableFuture resFuture = scheduledFuture.get();
return resFuture == null || resFuture.isCancelled();
} catch (Exception e) {
return false;
}
}
return false;
}
@Override
public boolean isDone() {
if (scheduledFuture.isDone()) {
try {
final CompletableFuture resFuture = scheduledFuture.get();
return resFuture == null || resFuture.isDone();
} catch (Exception e) {
return false;
}
}
return false;
}
@Override
public T get() throws InterruptedException, ExecutionException {
final CompletableFuture resFuture = scheduledFuture.get();
return resFuture == null ? null : resFuture.get();
}
@Override
public T get(long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException {
final long beginTime = DateUtil.currentMillis();
final CompletableFuture resFuture = scheduledFuture.get(timeout, unit);
final long remainingTimeout = unit.toMillis(timeout) - (DateUtil.currentMillis() - beginTime);
return resFuture == null ? null : (remainingTimeout > 0 ? resFuture.get(remainingTimeout, TimeUnit.MILLISECONDS) : resFuture.get());
}
};
}
static final class _UIExecutor implements Executor {
private static final Handler HANDLER = new Handler(Looper.getMainLooper());
private _UIExecutor() {
// Singleton.
}
@Override
public void execute(Runnable command) {
HANDLER.post(command);
}
public void execute(Runnable command, final long delay) {
if (delay > 0) {
HANDLER.postDelayed(command, delay);
} else {
HANDLER.post(command);
}
}
}
public static final class SerialExecutor {
private SerialExecutor() {
// singleton
}
/**
* The action will be asynchronously executed with {@code android.io.AsyncTask#SERIAL_EXECUTOR} in background.
*
* @param action
* @return
*/
public static CompletableFuture execute(final Runnable action) {
return Async.execute(action);
}
public static CompletableFuture execute(final Runnable action, final long delay) {
return Async.execute(action, delay);
}
public static CompletableFuture execute(final Try.Runnable extends Exception> action, final int retryTimes, final long retryInterval,
final Predicate super Exception> retryCondition) {
return Async.execute(action, retryTimes, retryInterval, retryCondition);
}
/**
* The action will be asynchronously executed with {@code android.io.AsyncTask#SERIAL_EXECUTOR} in background.
*
* @param action
* @return
*/
public static CompletableFuture execute(final Callable action) {
return Async.execute(action);
}
public static CompletableFuture execute(final Callable action, final long delay) {
return Async.execute(action, delay);
}
public static CompletableFuture execute(final Callable action, final int retryTimes, final long retryInterval,
final BiPredicate super T, ? super Exception> retryCondition) {
return Async.execute(action, retryTimes, retryInterval, retryCondition);
}
}
public static final class TPExecutor {
private TPExecutor() {
// singleton
}
/**
* The action will be asynchronously executed with {@code android.io.AsyncTask#THREAD_POOL_EXECUTOR} in background.
*
* @param action
* @return
*/
public static CompletableFuture execute(final Runnable action) {
return Async.executeWithThreadPool(action);
}
public static CompletableFuture execute(final Runnable action, final long delay) {
return Async.executeWithThreadPool(action, delay);
}
public static CompletableFuture execute(final Try.Runnable extends Exception> action, final int retryTimes, final long retryInterval,
final Predicate super Exception> retryCondition) {
return Async.executeWithThreadPool(action, retryTimes, retryInterval, retryCondition);
}
/**
* The action will be asynchronously executed with {@code android.io.AsyncTask#THREAD_POOL_EXECUTOR} in background.
*
* @param action
* @return
*/
public static CompletableFuture execute(final Callable action) {
return Async.executeWithThreadPool(action);
}
public static CompletableFuture execute(final Callable action, final long delay) {
return Async.executeWithThreadPool(action, delay);
}
public static CompletableFuture execute(final Callable action, final int retryTimes, final long retryInterval,
final BiPredicate super T, ? super Exception> retryCondition) {
return Async.executeWithThreadPool(action, retryTimes, retryInterval, retryCondition);
}
}
public static final class UIExecutor {
private UIExecutor() {
// singleton
}
/**
* The action will be asynchronously executed in UI thread.
*
* @param action
* @return
*/
public static CompletableFuture execute(final Runnable action) {
return Async.executeOnUiThread(action);
}
/**
* The action will be asynchronously executed in UI thread.
*
* @param action
* @param delay unit is milliseconds
* @return
*/
public static CompletableFuture execute(final Runnable action, final long delay) {
return Async.executeOnUiThread(action, delay);
}
public static CompletableFuture execute(final Try.Runnable extends Exception> action, final int retryTimes, final long retryInterval,
final Predicate super Exception> retryCondition) {
return Async.executeOnUiThread(action, retryTimes, retryInterval, retryCondition);
}
/**
* The action will be asynchronously executed in UI thread.
*
* @param action
* @return
*/
public static CompletableFuture execute(final Callable action) {
return Async.executeOnUiThread(action);
}
/**
* The action will be asynchronously executed in UI thread.
*
* @param action
* @param delay
* @return
*/
public static CompletableFuture execute(final Callable action, final long delay) {
return Async.executeOnUiThread(action, delay);
}
public static CompletableFuture execute(final Callable action, final int retryTimes, final long retryInterval,
final BiPredicate super T, ? super Exception> retryCondition) {
return Async.executeOnUiThread(action, retryTimes, retryInterval, retryCondition);
}
}
// /**
// * Short name for AsyncExecutor
// *
// * @deprecated replaced with SerialExecutor/TPExecutor/UIExecutor.
// */
// @Deprecated
// @Beta
// public static class Asyn extends AsyncExecutor {
// private Asyn() {
// // singleton
// }
// }
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy