Spring支持使用注解开启定时任务和异步方法执行
为了支持@Scheduled 和 @Async 注解,需要在你的@Configuration类上加 @EnableScheduling
和@EnableAsync
注解
@EnableScheduling
@EnableAsync
@SpringBootApplication
public class SpringBootApiApplication {
public static void main(String[] args) {
SpringApplication.run(SpringBootApiApplication.class,args);
}
}
使用@Scheduled注解
@Component
public class HelloJob {
@Scheduled(cron = "0/10 * * * * * ")
public void hello(){
System.out.println("hello");
}
}
除了时间表达式,还有其他的方式,比如:
@Scheduled(fixedDelay=5000) ---固定延迟每5秒执行一次,从上次执行完成开始测量时间
@Scheduled(fixedRate=5000) ---两次任务执行时间间隔为5秒
使用@Async
可以在方法上使用@Async注解,以便异步调用该方法,调用者将在调用时立刻返回,实际的执行已经提交给Spring taskExecutor的任务中。
@Async
void doSomething() {
// this will be executed asynchronously
}
方法可使用参数
@Async
void doSomething(String s) {
// this will be executed asynchronously
}
还可以异步调用返回值的方法,这些方法需要具有Future类型的返回值。
@Async
Future returnSomething(int i) {
// this will be executed asynchronously
}
@Async不能与生命周期回调一起使用,例如@PostConstruct,如果要异步初始化Spring Bean,必须使用单独的Spring bean,然后在目标方法上使用@Async注解
public class SampleBeanImpl implements SampleBean {
@Async
void doSomething() {
// ...
}
}
public class SampleBeanInitializer {
private final SampleBean bean;
public SampleBeanInitializer(SampleBean bean) {
this.bean = bean;
}
@PostConstruct
public void initialize() {
bean.doSomething();
}
}
当@Async方法具有Future类型返回值时,很容易管理在方法执行期间抛出的异常,因为在调用get结果时会抛出此异常。 但是,对于void返回类型,异常未被捕获且无法传输。 对于这些情况,可以提供AsyncUncaughtExceptionHandler来处理此类异常。
public class MyAsyncUncaughtExceptionHandler implements AsyncUncaughtExceptionHandler {
@Override
public void handleUncaughtException(Throwable ex, Method method, Object... params) {
// handle exception
}
}