轉載請注明來源 賴賴的博客
導語
一個簡單的抽象,可以解決百分之八十的問題。
在寫Spring 應用的時候,會遇到一些異步執行和任務調度的問題,例如:Spring MVC中需要向微信發送請求,告訴微信進行公眾號推送。這個時候需要用到異步執行,而周報表、月報表、日最高點擊等,需要用到任務調度。
Spring高度凝聚了及其簡單的調度和異步執行方式,迅速解決百分之八十的問題。
實例
項目工程目錄結構和代碼獲取地址
獲取地址(版本Log將會注明每一個版本對應的課程)
https://github.com/laiyijie/SpringLearning
目錄結構
運行工程
運行具有Main函數的 App.java
得到如下輸出
the 2 time to say userHello
the 0 time to say userHello
the 3 time to say userHello
the 1 time to say userHello
the 4 time to say userHello
the 6 time to say userHello
the 5 time to say userHello
the 8 time to say userHello
the 9 time to say userHello
the 7 time to say userHello
time:1480341514063 userHello
time:1480341515064 userHello
time:1480341516065 userHello
time:1480341517066 userHello
time:1480341518067 userHello
time:1480341519068 userHello
time:1480341520069 userHello
time:1480341521070 userHello
time:1480341522071 userHello
項目詳解
從App.java入手
App.java
package me.laiyijie.demo;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import me.laiyijie.demo.service.HelloInterface;
public class App {
public static void main(String[] args) throws InterruptedException {
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("root-context.xml");
HelloInterface hello = context.getBean(HelloInterface.class);
for (int i = 0; i < 10; i++) {
hello.sayHello(i);
}
Thread.sleep(10000);
context.close();
}
}
主函數中有一個for循環,調用的是HelloInterface
的sayHello
方法。
并且主線程休眠了10秒后關閉ApplicationContext
。
HelloInterface.java
package me.laiyijie.demo.service;
public interface HelloInterface{
void sayHello(int i );
void sayHelloEverySecondes();
}
接口,并定義了兩個函數,其實現類為 UserServiceImpl
UserServiceImpl.java
package me.laiyijie.demo.service;
import org.springframework.scheduling.annotation.Async;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Service;
@Service
public class UserServiceImpl implements HelloInterface {
@Async
public void sayHello(int i) {
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("the " + i + " time to say " + "userHello");
}
@Scheduled(fixedDelay = 1000)
public void sayHelloEverySecondes() {
System.out.println("time:" + System.currentTimeMillis() + " userHello");
}
}
實現了兩個函數,而且兩個函數前面分別有@Async
和@Scheduled
兩個注解。
分別探討:
@Async 異步執行
被@Async
標注過的方法不是在主線程中執行,是另開了一個線程,并且進行調度,從打印就可以看出,調度是隨機的!
既然涉及到異步,就涉及到線程池有多大?隊列有多大?
root-context.xml中有如下配置:
<task:executor id="myExecutor" pool-size="5" queue-capacity="100"/>
pool-size=5 : 線程池的大小為5!
queue-capacity=100 : 等待隊列的最大長度為100!
所以可以看到,前面的輸出,0-4次userHello
在5-9次前面,因為一次只能同時執行五個線程。
@Scheduled(fixedDelay = 1000)
被@Scheduled
標注過的函數會按照配置的運行方式自動執行
!此處配置的是fixedDelay=1000
含義是每隔1000ms執行一次(在上次執行完后開始計時1000ms)
定時調度一樣有一個線程池:
<task:scheduler id="myScheduler" pool-size="5"/>
含義與executor
一致
需要開啟@Async
和@Scheduled
,root-context.xml需要進行如下配置:
root-context.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:task="http://www.springframework.org/schema/task"
xsi:schemaLocation="http://www.springframework.org/schema/task http://www.springframework.org/schema/task/spring-task-4.3.xsd
http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.3.xsd
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.3.xsd">
<context:component-scan base-package="me.laiyijie.demo"></context:component-scan>
<task:annotation-driven executor="myExecutor" scheduler="myScheduler" />
<task:scheduler id="myScheduler" pool-size="5"/>
<task:executor id="myExecutor" pool-size="5" queue-capacity="100"/>
</beans>
變更如下:
-
增加
task
命名空間xmlns:task="http://www.springframework.org/schema/task"
-
增加注解驅動以及執行器
<task:annotation-driven executor="myExecutor" scheduler="myScheduler" /> <task:scheduler id="myScheduler" pool-size="5"/> <task:executor id="myExecutor" pool-size="5" queue-capacity="100"/>
雖然可以最簡單的使用:
<task:annotation-driven />
但是建議還是自己定義executor
和scheduler
方便控制資源大小
pom.xml
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>me.laiyijie</groupId>
<artifactId>demo</artifactId>
<version>0.0.1-SNAPSHOT</version>
<packaging>jar</packaging>
<dependencies>
<!-- https://mvnrepository.com/artifact/org.springframework/spring-context -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>4.3.2.RELEASE</version>
</dependency>
</dependencies>
</project>
小結
- 異步執行使用
@Async
標注方法 - 定時任務使用
@Scheduled
標注方法