>  기사  >  Java  >  Spring의 @Async 사용법에 대한 자세한 설명과 간단한 예제 소개

Spring의 @Async 사용법에 대한 자세한 설명과 간단한 예제 소개

黄舟
黄舟원래의
2017-03-07 10:12:131629검색

본 글에서는 Spring의 @Async 사용법에 대한 자세한 설명과 간단한 예시를 주로 소개합니다. 필요한 친구들은

Spring의 @Async 사용법

을 참고하세요. : 대부분의 Java 애플리케이션에서는 대부분의 경우 동기화를 통해 상호 작용 처리가 구현되지만 타사 시스템과 상호 작용할 때 응답 속도가 느려지기 쉽습니다. 과거에는 이러한 작업을 완료하는 데 대부분이 사용되었습니다. 실제로 Spring 3.x 이후에는 이 문제를 완벽하게 해결하기 위해 @Async가 내장되었습니다. 이 기사에서는 @Async의 사용법을 완전히 소개합니다.

1. 비동기 호출이란?

비동기 호출을 설명하기 전에 먼저 동기 호출의 정의를 살펴보겠습니다. 동기화란 전체 처리 프로세스가 순차적으로 실행되고 각 프로세스가 완료되면 결과가 반환되는 것을 의미합니다. 비동기 호출은 호출 명령만 전송하며, 호출자는 호출된 메서드가 완전히 실행될 때까지 기다릴 필요 없이 다음 프로세스를 계속 실행합니다.

예를 들어 특정 호출에서 세 가지 프로세스 메서드 A, B, C를 순차적으로 호출해야 하며, 모두 동기 호출인 경우 프로세스가 완료된 것으로 간주되기 전에 순차적으로 실행되어야 합니다. B가 비동기 호출 방법인 경우 A를 실행한 후 B를 호출하면 B가 완료될 때까지 기다리지 않고 실행을 시작하고 C를 호출합니다. C가 실행된 후 프로세스가 완료되었음을 의미합니다.

2. 기존 비동기 호출 처리 방법

Java에서는 유사한 시나리오를 처리할 때 일반적으로 해당 비동기 호출 논리 패스를 완료하기 위해 독립적인 스레드를 생성하는 방식을 사용합니다. 메인 스레드와 다른 스레드 사이의 실행 프로세스를 통해 독립 스레드를 시작한 후에도 메인 스레드는 지연 및 대기 없이 계속 실행됩니다.

3. @Async 소개

Spring에서는 @Async 주석을 기반으로 하는 메서드를 비동기 메서드라고 하며, 이러한 메서드는 별도로 실행됩니다. 스레드의 경우 호출자는 다른 작업을 계속하기 전에 완료될 때까지 기다릴 필요가 없습니다.

Spring에서 @Async를 활성화하는 방법

Java 구성 기반 활성화 방법:

@Configuration 
@EnableAsync 
public class SpringAsyncConfig { ... }

XML 구성 파일 기반 활성화 방법은 다음과 같이 구성됩니다.

<task:executor id="myexecutor" pool-size="5" /> 
<task:annotation-driven executor="myexecutor"/>

위의 두 가지 정의 방법은 다음과 같습니다.

4. @Async 기반으로 반환 값 없이

호출 예시는 다음과 같습니다.

@Async //标注使用 
public void asyncMethodWithVoidReturnType() { 
  System.out.println("Execute method asynchronously. " 
   + Thread.currentThread().getName()); 
}

사용된 방법은 매우 간단합니다. 하나의 레이블로 모든 문제를 해결할 수 있습니다.

5. @Async 반환 값을 기준으로 호출

예시는 다음과 같습니다.

@Async 
public Future<String> asyncMethodWithReturnType() { 
  System.out.println("Execute method asynchronously - " 
   + Thread.currentThread().getName()); 
  try { 
    Thread.sleep(5000); 
    return new AsyncResult<String>("hello world !!!!"); 
  } catch (InterruptedException e) { 
    // 
  } 
  
  return null; 
}

위의 예시를 보면 반환된 데이터 타입이 인터페이스인 Future 타입임을 알 수 있습니다. 구체적인 결과 유형은 주의가 필요한 AsyncResult입니다.

결과를 ​​반환하는 비동기 메소드 호출 예:

public void testAsyncAnnotationForMethodsWithReturnType() 
  throws InterruptedException, ExecutionException { 
  System.out.println("Invoking an asynchronous method. " 
   + Thread.currentThread().getName()); 
  Future<String> future = asyncAnnotationExample.asyncMethodWithReturnType(); 
  
  while (true) { ///这里使用了循环判断,等待获取结果信息 
    if (future.isDone()) { //判断是否执行完毕 
      System.out.println("Result from asynchronous process - " + future.get()); 
      break; 
    } 
    System.out.println("Continue doing something else. "); 
    Thread.sleep(1000); 
  } 
}

분석: 이러한 비동기 메소드의 결과 정보를 얻습니다. by 지속적으로 현재 비동기 메소드가 실행되었는지 여부를 얻기 위해 Future 상태를 확인하여 구현됩니다.

6. @Async 호출 기반 예외 처리 메커니즘

비동기 방식에서는 예외가 발생하면 호출자가 Perceived할 수 없습니다. 예외 처리가 실제로 필요한 경우 다음과 같이 처리합니다.

1. AsyncTaskExecutor를 구현하는 작업 실행기를 사용자 정의합니다.

여기에서 특정 예외를 처리하는 논리와 방법을 정의합니다.

2. 내장된 작업 실행기를 대체하도록 사용자 지정 TaskExecutor 구성

예시 1단계, 사용자 지정 TaskExecutor

public class ExceptionHandlingAsyncTaskExecutor implements AsyncTaskExecutor { 
  private AsyncTaskExecutor executor; 
  public ExceptionHandlingAsyncTaskExecutor(AsyncTaskExecutor executor) { 
    this.executor = executor; 
   } 
   ////用独立的线程来包装,@Async其本质就是如此 
  public void execute(Runnable task) {    
   executor.execute(createWrappedRunnable(task)); 
  } 
  public void execute(Runnable task, long startTimeout) { 
    /用独立的线程来包装,@Async其本质就是如此 
    executor.execute(createWrappedRunnable(task), startTimeout);      
  }  
  public Future submit(Runnable task) { return executor.submit(createWrappedRunnable(task)); 
    //用独立的线程来包装,@Async其本质就是如此。 
  }  
  public Future submit(final Callable task) { 
   //用独立的线程来包装,@Async其本质就是如此。 
    return executor.submit(createCallable(task));  
  }  
   
  private Callable createCallable(final Callable task) {  
    return new Callable() {  
      public T call() throws Exception {  
         try {  
           return task.call();  
         } catch (Exception ex) {  
           handle(ex);  
           throw ex;  
          }  
         }  
    };  
  } 
 
  private Runnable createWrappedRunnable(final Runnable task) {  
     return new Runnable() {  
       public void run() {  
         try { 
           task.run();  
         } catch (Exception ex) {  
           handle(ex);  
          }  
      } 
    };  
  }  
  private void handle(Exception ex) { 
   //具体的异常逻辑处理的地方 
   System.err.println("Error during @Async execution: " + ex); 
  } 
}

분석: 독립적인 스레드를 사용하여 특정 메소드 작업을 수행하는 AsyncTaskExecutor를 구현하는 것을 확인할 수 있습니다. createCallable 및 createWrapperRunnable에는 예외 처리 방법 및 메커니즘이 정의되어 있습니다.

handle()은 앞으로 예외 처리에 집중해야 할 부분입니다.

구성 파일의 내용:

<task:annotation-driven executor="exceptionHandlingTaskExecutor" scheduler="defaultTaskScheduler" /> 
<bean id="exceptionHandlingTaskExecutor" class="nl.jborsje.blog.examples.ExceptionHandlingAsyncTaskExecutor"> 
  <constructor-arg ref="defaultTaskExecutor" /> 
</bean> 
<task:executor id="defaultTaskExecutor" pool-size="5" /> 
<task:scheduler id="defaultTaskScheduler" pool-size="1" />

분석: 여기의 구성은 기본 TaskExecutor를 대체하기 위해 사용자 정의 taskExecutor를 사용합니다.

7. @Async 호출의 트랜잭션 처리 메커니즘

@Async로 표시된 메소드에는 @Transactional도 표시됩니다. , 비동기 처리 기반의 작업이므로 트랜잭션 관리 제어가 불가능합니다.

그렇다면 이러한 작업에 트랜잭션 관리를 추가하는 방법은 무엇일까요? 트랜잭션 관리 작업이 필요한 메서드는 비동기식 메서드 내에 배치할 수 있으며 내부에서 호출되는 메서드에는 @Transactional.

이 추가됩니다. 예: 메서드 A는 @Async/@Transactional로 표시되지만 트랜잭션은 그렇지 않습니다. 제어 목적으로 생성됩니다.

메소드 B에는 @Async라는 주석이 붙고, C와 D에는 B에서 호출됩니다. C/D에는 각각 @Transactional이라는 주석이 붙어 있어 트랜잭션 제어 목적을 달성할 수 있습니다.

8. 요약

위의 설명을 통해 @Async를 사용하는 방법과 주의사항을 알아야 합니다.

위 내용은 Spring에서의 @Async 사용법에 대한 자세한 설명과 간단한 예시입니다. 더 많은 관련 내용은 PHP 중국어 홈페이지(www.php.cn)를 참고해주세요!


성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.