06 Spring 異步執行,任務調度(@Schedule、@Async)

轉載請注明來源 賴賴的博客

導語

一個簡單的抽象,可以解決百分之八十的問題。

在寫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循環,調用的是HelloInterfacesayHello方法。
并且主線程休眠了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 />   

但是建議還是自己定義executorscheduler方便控制資源大小

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>  

小結

  1. 異步執行使用 @Async標注方法
  2. 定時任務使用@Scheduled 標注方法
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容