Spring boot 注解@Async无效,不起作用

 

 

springboot1.5+,项目框架中集成了异步多线程操作配置..

同类方法中使用异步注解没有作用,使用小的demo做了一个样例......,这样执行后就,是串行单线程......

package com.dianshichengjin.makemoney.web.rest.controller;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.scheduling.annotation.Async;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

/**
 * Created by Administrator on 2019/3/11.
 * 描述: 测试异步调用;
 * 测试多线程调用
 *
 * @author Young
 * @create 2019-03-11 16:27
 */
@RestController
@RequestMapping("/")
public class AsyncController {

    private  final Logger log =  LoggerFactory.getLogger(this.getClass());

    @Autowired
    private AsyncControllerTest asyncControllerTest;

    /**
     * @Author Young
     * @Description //方法调用接口类
     * @Date 16:46 2019/3/11
     * @param
     * @return java.lang.String
     **/
    @GetMapping("test")
    public String test(){
//        asyncControllerTest.doAsyncMethod();
//        asyncControllerTest.doAsyncMethod();
        doAsyncMethod();
        doAsyncMethod();
        log.debug("主线程执行结束......");
        return "SUCCESS";
    }

    /**
     * @Author Young
     * @Description //同类方法中的异步标签
     * @Date 16:45 2019/3/11
     * @param
     * @return void
     **/
    @Async
    private void doAsyncMethod() {
        try {
            //todo
            Thread.sleep(2000);
            log.debug("异步方法执行了......");
        }catch (Exception e){
            e.printStackTrace();
        }
    }

}

 

修改后,

 

改正使用调用另外的类,然后通过注入的方式注入对象:

 

package com.dianshichengjin.makemoney.web.rest.controller;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.scheduling.annotation.Async;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

/**
 * Created by Administrator on 2019/3/11.
 * 描述: 测试异步调用;
 * 测试多线程调用
 *
 * @author Young
 * @create 2019-03-11 16:27
 */
@RestController
@RequestMapping("/")
public class AsyncController {

    private  final Logger log =  LoggerFactory.getLogger(this.getClass());

    @Autowired
    private AsyncControllerTest asyncControllerTest;

    /**
     * @Author Young
     * @Description //方法调用接口类
     * @Date 16:46 2019/3/11
     * @param
     * @return java.lang.String
     **/
    @GetMapping("test")
    public String test(){
        asyncControllerTest.doAsyncMethod();
        asyncControllerTest.doAsyncMethod();
//        doAsyncMethod();
//        doAsyncMethod();
        log.debug("主线程执行结束......");
        return "SUCCESS";
    }

    /**
     * @Author Young
     * @Description //同类方法中的异步标签
     * @Date 16:45 2019/3/11
     * @param
     * @return void
     **/
    @Async
    private void doAsyncMethod() {
        try {
            //todo
            Thread.sleep(2000);
            log.debug("异步方法执行了......");
        }catch (Exception e){
            e.printStackTrace();
        }
    }

}

 

package com.dianshichengjin.makemoney.web.rest.controller;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Component;

/**
 * Created by Administrator on 2019/3/11.
 * 描述:
 *
 * @author Young
 * @create 2019-03-11 16:39
 */
@Component
public class AsyncControllerTest {


    private  final Logger log =  LoggerFactory.getLogger(this.getClass());

    @Async
    public void doAsyncMethod() {
        try {
            //todo
            Thread.sleep(2000);
            log.debug("异步方法执行了......");
        }catch (Exception e){
            e.printStackTrace();
        }
    }
}

 

 

究其原因:

 

//TODO

简单说就是没有过去到代理类,本类调用时,直接自己内部调用,没有走代理类..........

失效原因

1.没有在@SpringBootApplication启动类当中添加注解@EnableAsync注解。
2.异步方法使用注解@Async的返回值只能为void或者Future。
3.没有走Spring的代理类。因为@Transactional和@Async注解的实现都是基于Spring的AOP,而AOP的实现是基于动态代理模式实现的。那么注解失效的原因就很明显了,有可能因为调用方法的是对象本身而不是代理对象,因为没有经过Spring容器。

第二点和第三点容易犯......

 

解决方法:

这里具体说一下第三种情况的解决方法。
1.注解的方法必须是public方法。
2.方法一定要从另一个类中调用,也就是从类的外部调用,类的内部调用是无效的。
3.如果需要从类的内部调用,需要先获取其代理类,下面上代码

@Service
public class XxxService{
  public void methodA(){
    ...
    XxxService xxxServiceProxy = SpringUtil.getBean(XxxService.class);
    xxxServiceProxy.methodB();
    ...
  }

  @Async
  public void methodB() {
    ...
  }
}

 

很牛逼的样子........

本类中可以定义实现本类中调用的本类的异步多线程方法;

必须实现的两种:

  • public方法
  • 手动获取spring bean

本类执行private的方法中,异步执行报错。

修改为public方法可用。

 

故,正确方式为:

 

package com.zhiliyuchi.web.rest.controller;

import com.zhiliyuchi.domain.fishpond.ManagerEntity;
import com.zhiliyuchi.repository.fishpond.ManagerRepository;
import com.zhiliyuchi.web.rest.util.Result;
import com.zhiliyuchi.web.rest.util.SpringUtils;
import org.aspectj.lang.annotation.Aspect;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.scheduling.annotation.Async;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import java.util.List;

/**
 * @创建人:Young
 * @时 间: 2019/3/13
 * @描 述: Manager类Controller接口类
 */
@RestController
@RequestMapping("manager")
public class ManagerController {
    private final Logger logger = LoggerFactory.getLogger(ManagerController.class);
    @Autowired
    private ManagerRepository managerRepository;
//    @Autowired
//    private AsyncController asyncController;

    /**
     * @描述 查询所有的数据的异常测试类
     * @参数  []
     * @返回值  com.zhiliyuchi.web.rest.util.Result
     * @创建人  Young
     * @创建时间  2019/3/13 14:05
     **/
    @GetMapping("all")
    public Result getAll(){
        ManagerController bean = SpringUtils.getBean(ManagerController.class);
        logger.debug("查询所有的manager");
        List<ManagerEntity> all = managerRepository.findAll();
        bean.test();
        bean.test();
        return Result.error("数据请求服务未启用","type","subtype");
    }


    @Async
    public void test(){
        try {
            logger.debug("=======开始执行异步处理类========");
            Thread.sleep(2000);
        }catch (Exception e){
            e.printStackTrace();
        }
    }




}

SpringUtils的工具类,手动获取bean方法:

package com.zhiliyuchi.web.rest.util;

import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;

/**
 * @创建人:Young
 * @时 间: 2019/3/13
 * @描 述: 手动获取spring的bean
 */
@Component("springContextUtil")
public class SpringUtils implements ApplicationContextAware {


    private static ApplicationContext applicationContext = null;

    public static ApplicationContext getApplicationContext() {
        return applicationContext;
    }

    @SuppressWarnings("unchecked")
    public static <T> T getBean(String beanId) {
        return (T) applicationContext.getBean(beanId);
    }

    public static <T> T getBean(Class<T> requiredType) {
        return (T) applicationContext.getBean(requiredType);
    }
    /**
     * Spring容器启动后,会把 applicationContext 给自动注入进来,然后我们把 applicationContext
     *  赋值到静态变量中,方便后续拿到容器对象
     * @see org.springframework.context.ApplicationContextAware#setApplicationContext(org.springframework.context.ApplicationContext)
     */
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        SpringUtils.applicationContext = applicationContext;
    }
}

 

@Transactional 加于private方法, 无效
@Transactional 加于未加入接口的public方法, 再通过普通接口方法调用, 无效
@Transactional 加于接口方法, 无论下面调用的是private或public方法, 都有效
@Transactional 加于接口方法后, 被本类普通接口方法直接调用, 无效
@Transactional 加于接口方法后, 被本类普通接口方法通过接口调用, 有效
@Transactional 加于接口方法后, 被它类的接口方法调用, 有效
@Transactional 加于接口方法后, 被它类的私有方法调用后, 有效
 

 

 

end;

  • 51
    点赞
  • 86
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
好的,下面是一个简单的示例: 假设我们有一个 UserService,其中有一个方法 sendEmail,它需要异步地发送电子邮件。现在我们想要将当前用户的信息透传给异步任务中使用的线程。 首先,我们需要在异步方法上添加 @Async 注解,并在配置类中启用异步支持: ```java @Configuration @EnableAsync public class AppConfig implements AsyncConfigurer { @Override public Executor getAsyncExecutor() { ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor(); executor.setCorePoolSize(10); executor.setMaxPoolSize(100); executor.setQueueCapacity(10); executor.initialize(); return executor; } @Override public AsyncUncaughtExceptionHandler getAsyncUncaughtExceptionHandler() { return new CustomAsyncExceptionHandler(); } } ``` 在上面的示例中,我们创建了一个 ThreadPoolTaskExecutor,它将用于执行异步任务。我们还实现了 AsyncConfigurer 接口,并覆盖了 getAsyncExecutor 和 getAsyncUncaughtExceptionHandler 方法,以提供自定义的 Executor 和异常处理程序。 现在我们需要将当前用户信息存储在一个 ThreadLocal 对象中。这可以通过一个拦截器来实现: ```java public class UserContextInterceptor extends HandlerInterceptorAdapter { private final ThreadLocal<String> userThreadLocal = new ThreadLocal<>(); @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { String currentUser = request.getHeader("X-User"); userThreadLocal.set(currentUser); return true; } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { userThreadLocal.remove(); } public String getCurrentUser() { return userThreadLocal.get(); } } ``` 在上面的示例中,我们创建了一个 UserContextInterceptor,它将在每个请求的开始和结束时执行。在 preHandle 方法中,我们从请求头中获取当前用户信息,并将其存储在一个 ThreadLocal 对象中。在 afterCompletion 方法中,我们将删除该信息,以避免内存泄漏。 现在,我们可以在 UserService 的 sendEmail 方法中使用 UserContextInterceptor 中存储的当前用户信息: ```java @Service public class UserService { @Autowired private JavaMailSender mailSender; @Autowired private UserContextInterceptor userContextInterceptor; @Async public void sendEmail(String to, String subject, String text) { String currentUser = userContextInterceptor.getCurrentUser(); // 使用当前用户信息发送电子邮件 // ... } } ``` 在上面的示例中,我们使用 @Autowired 注解将 UserContextInterceptor 注入到 UserService 中。在 sendEmail 方法中,我们从 UserContextInterceptor 中获取当前用户信息,并在发送电子邮件时使用它。 通过这种方式,我们可以将当前用户信息透传给异步任务中使用的线程。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值