spring 异步方法(@Async注解代替多线程 )

最近在开发过程里遇到让人很头痛的功能,就是一个批量复制功能,批量复制中包括数据库中的数据,还有文件系统的复制。这在开发中要考虑到系统性能和友好度的问题,一个批量复制最少要执行1~3分钟,这让用户在点击一个按钮后要等待1~3分钟不现实,最后只能用多线程,来达到用户的友好度。

 

在项目既然用到了Spring ,我们用Spring实现的多线程来实现这个功能。

 

 

public class JobUtilsTest{  
      
    @Autowired  
    private DaoService service;  
  
    @Test  
    public void testAsync() throws Exception {  
        System.out.println("start" );  
        service.update(); // ★ 假设这个方法会比较耗时,需要异步执行  
        System.out.println("end");  
          
          
        Thread.sleep(3000); // 因为junit结束会结束jvm,所以让它等会异步线程  
    }  
}  

 

 

在方法上加@Async注解 这个方法就成异步方法了

异步方法不能和调用方法放在一个类里面

 

@Service  
public class DaoService {  
   @Async
    public void update() {  
        try {  
            Thread.sleep(2000);  
            // do something  
        } catch (InterruptedException e) {  
            e.printStackTrace();  
        }  
        System.out.println("operation complete.");  
    }  
}  


applicationContext.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:p="http://www.springframework.org/schema/p"  
    xmlns:context="http://www.springframework.org/schema/context"  
    xmlns:mvc="http://www.springframework.org/schema/mvc" xmlns:task="http://www.springframework.org/schema/task"  
    xsi:schemaLocation="http://www.springframework.org/schema/beans   
    http://www.springframework.org/schema/beans/spring-beans-3.0.xsd  
    http://www.springframework.org/schema/context   
    http://www.springframework.org/schema/context/spring-context-3.0.xsd  
    http://www.springframework.org/schema/task http://www.springframework.org/schema/task/spring-task-3.0.xsd">  
  
    <context:component-scan base-package="com.chinacache" />
    <!--Spring 的配置文件中一定要配置这一项 -->
    <task:annotation-driven/> 
</beans>

输出结果:
start
end
operation complete.

 

 

 

Spring boot 配置

配置线程池

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.aop.interceptor.AsyncUncaughtExceptionHandler;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.annotation.AsyncConfigurer;
import org.springframework.scheduling.annotation.EnableAsync;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;

import java.lang.reflect.Method;
import java.util.concurrent.Executor;

@Configuration
@EnableAsync
public class AsyncConfig  implements AsyncConfigurer {

    private static final Logger LOGGER = LoggerFactory.getLogger(AsyncConfig.class);

    @Override
    public Executor getAsyncExecutor() {
        ThreadPoolTaskExecutor taskExecutor = new ThreadPoolTaskExecutor();
        taskExecutor.setCorePoolSize(5);
        taskExecutor.setMaxPoolSize(6);
        taskExecutor.setQueueCapacity(1024);
        taskExecutor.initialize();
        return taskExecutor;
    }

    @Override
    public AsyncUncaughtExceptionHandler getAsyncUncaughtExceptionHandler() {
        return new AsyncUncaughtExceptionHandler() {
            @Override
            public void handleUncaughtException(Throwable ex, Method method, Object... params) {
                LOGGER.warn("Exception: message - {},Method name - {}", ex.getMessage(), method.getName(), ex);
                for (Object param : params) {
                    LOGGER.warn("Exception: Parameter value - {}", param);
                }
            }
        };
    }
}

使用  

注意: 调用者和 异步方法不能放在一个类中

@Service  
public class DaoService {  
   @Async
    public void update() {  
        try {  
            Thread.sleep(2000);  
            // do something  
        } catch (InterruptedException e) {  
            e.printStackTrace();  
        }  
        System.out.println("operation complete.");  
    }  
}  

 

 

 

 

 

  • 0
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 使用@async注解可以在方法调用时将其作为异步任务执行。而在@async注解内调用feign,则可以实现异步的远程调用服务。 在使用@async注解时,需要在方法前加上该注解,并返回Future或CompletableFuture对象。然后在方法体内,可以使用feign来调用远程服务,实现异步的远程服务调用。 使用@async注解内调用feign可以提高程序并发性能,减少线程阻塞等待远程服务响应的时间。同时,由于使用了异步调用,系统可以更加容易地扩展到多个服务实例,更好地支持高并发和负载均衡。 需要注意的是,使用@async注解内调用feign需要确保异步任务不会阻塞主线程。因此,在调用方法时需要使用合适的线程池,以避免线程池耗尽的情况发生。此外,需要仔细评估异步调用对于整个系统的影响,以确保异步调用不会影响系统稳定性和可靠性。 总之,使用@async注解内调用feign可以提高程序的性能和可扩展性,但需要仔细权衡异步调用的影响,并确保代码的正确性和稳定性。 ### 回答2: @async注解Spring Boot中的一个注解,它指示Spring Boot应该在另一个线程中异步执行方法调用。Feign是一个轻量级的HTTP客户端库,用于将RESTful服务作为接口进行访问。在Spring Boot应用程序中使用Feign库时,可以使用@FeignClient注解将Feign客户端绑定到REST服务。在异步调用Feign库之前,必须在Spring Boot应用程序中配置异步分组。在此之后,就可以在@async注解中调用Feign库了。 具体实现步骤如下: 首先,需要为Spring Boot应用程序配置异步编组。这可以通过在定义的配置类上添加@EnableAsync和@Configuration注解来完成。这样,Spring Boot应用程序就能够在不同的线程中执行方法调用。 然后,在定义的Feign客户端接口上添加@FeignClient注解,并指定REST服务的名称。接着在该接口上定义异步方法,这些方法必须添加@async注解,以通知Spring Boot应该异步执行这些方法。最后,调用异步方法时,必须通过Feign库向REST服务发出请求,以获取所需的数据。 使用@async注解调用Feign库非常简单,只需要在Spring Boot应用程序中进行正确的配置,并使用Feign客户端接口定义异步方法即可。这样,应用程序就能在不同的线程中异步执行Feign调用,从而提高应用程序的响应性和性能。 ### 回答3: @async注解是用于异步调用方法注解,在调用方法时会开启一个新的线程来执行方法体中的代码,从而不会阻塞主线程。而Feign是一个声明式的Web Service客户端,可以让我们更容易地调用其它微服务。 在使用@async注解内调用Feign时,需要注意以下几点: 1. 定义Feign客户端并注入到调用方法所在的类中。 2. 在调用方法中使用@autowired注解将Feign客户端注入进来。 3. 在调用方法中使用@async注解,表示异步调用。 例如,下面是一个使用@async注解内调用Feign的示例代码: ``` @Service public class DemoService { @Autowired private OtherServiceClient otherServiceClient; @Async public void callOtherService() { otherServiceClient.doSomething(); } } @FeignClient(name = "other-service", url = "http://localhost:8001") public interface OtherServiceClient { @RequestMapping(value = "/do-something", method = RequestMethod.GET) void doSomething(); } ``` 在上面的代码中,DemoService中有一个名为callOtherService的方法,使用了@async注解,表示该方法是一个异步方法。另外,DemoService中还使用了@Autowired注解将OtherServiceClient注入进来。 OtherServiceClient则是一个Feign客户端,使用@FeignClient注解将其定义为一个Feign客户端,其中name表示服务名,url表示服务的URL地址。 在callOtherService方法中,调用了OtherServiceClient的doSomething方法,表示要调用other-service服务的/do-something接口。由于callOtherService使用了@async注解,因此doSomething方法的调用将会在一个新的线程中执行,不会阻塞主线程。 总的来说,使用@async注解内调用Feign可以提高代码的并发能力,避免因为调用其他服务而阻塞主线程。同时由于Feign是一个声明式的客户端,使得我们调用其他服务也变得更加简单。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值