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

io.github.icodegarden.nutrient.lang.schedule.AbstractSchedule Maven / Gradle / Ivy

The newest version!
package io.github.icodegarden.nutrient.lang.schedule;

import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.ScheduledThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;

import org.springframework.util.Assert;

import io.github.icodegarden.nutrient.lang.util.CronUtils;
import io.github.icodegarden.nutrient.lang.util.LogUtils;
import io.github.icodegarden.nutrient.lang.util.ThreadUtils;
import lombok.extern.slf4j.Slf4j;

/**
 * 
 * @author Fangfang.Xu
 *
 */
@Slf4j
public abstract class AbstractSchedule implements Schedule {

	private final String name;
	
	private final ScheduledThreadPoolExecutor scheduleThreadPool;

	private long scheduleTimes;

	private final AtomicBoolean started = new AtomicBoolean(false);
	private final AtomicBoolean closed = new AtomicBoolean(false);

	private ScheduledFuture future;

	public AbstractSchedule() {
		this.name = AbstractSchedule.this.getClass().getSimpleName();
		
		scheduleThreadPool = ThreadUtils.newSingleScheduledThreadPool(name);
		scheduleThreadPool.setRemoveOnCancelPolicy(true);
	}

	public AbstractSchedule(String name) {
		this.name = name;
		
		scheduleThreadPool = ThreadUtils.newSingleScheduledThreadPool(name);
		scheduleThreadPool.setRemoveOnCancelPolicy(true);
	}
	
	public String getName() {
		return name;
	}
	
	public long getScheduleTimes() {
		return scheduleTimes;
	}

	@Override
	public boolean scheduleWithFixedDelay(long initialDelayMillis, long scheduleMillis) {
		if (started.compareAndSet(false, true)) {
			future = scheduleThreadPool.scheduleWithFixedDelay(() -> {
				scheduling();
			}, initialDelayMillis, scheduleMillis, TimeUnit.MILLISECONDS);

			return true;
		}
		return false;
	}

	@Override
	public boolean scheduleAtFixedRate(long initialDelayMillis, long scheduleMillis) {
		if (started.compareAndSet(false, true)) {
			future = scheduleThreadPool.scheduleAtFixedRate(() -> {
				scheduling();
			}, initialDelayMillis, scheduleMillis, TimeUnit.MILLISECONDS);

			return true;
		}
		return false;
	}

	@Override
	public boolean scheduleWithCron(String cron) {
		Assert.isTrue(CronUtils.isValid(cron), "Invalid:cron");

		if (started.compareAndSet(false, true)) {

			doCron(cron);

			return true;
		}
		return false;
	}

	private void doCron(String cron) {
		future = scheduleThreadPool.schedule(() -> {
			scheduling();

			doCron(cron);
		}, CronUtils.nextDelayMillis(cron), TimeUnit.MILLISECONDS);
	}

	private void scheduling() {
		synchronized (this) {// 关闭等待用
			try {
				if (log.isInfoEnabled()) {
					log.info("{} schedule run, scheduleTimes:{}", this.getClass().getSimpleName(), scheduleTimes);
				}

				if (isClosed()) {
					log.info("{} schedule was closed, stop", this.getClass().getSimpleName());
					/**
					 * 如果已关闭,终止执行
					 */
					return;
				}
				doSchedule();
			} catch (Throwable e) {
				log.error("ex on {}", AbstractSchedule.this.getClass().getSimpleName(), e);
			} finally {
				scheduleTimes++;
			}
		}
	}

	protected abstract void doSchedule() throws Throwable;

	@Override
	public boolean isClosed() {
		return closed.get();
	}

	/**
	 * 阻塞直到处理完毕,这不会阻塞很久
	 */
	@Override
	public void close() {
		LogUtils.infoIfEnabled(log, () -> log.info("{} start close.", getName()));

		if (future != null) {
			future.cancel(false);//不要使用true,会使执行中的调度由于需要获取连接池中的连接抛出InterruptedException
		}
		closed.set(true);
		
		scheduleThreadPool.shutdown();//这一步是必须的,但以上future.cancel却不是必须的

		/**
		 * 使用synchronized保障如果任务正在处理中,则等待任务处理完毕
		 */
		synchronized (this) {
		}

		LogUtils.infoIfEnabled(log, () -> log.info("{} complete close.", getName()));
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy