多线程时Autowired自动注入问题

简介: 多线程时Autowired自动注入问题

首先需要知道的是,容器中的组件,也就是你添加了诸如@Component , @Service , @Controller以及@Repository等等注解,在容器启动的时候是会扫描标注这些注解的类创建bean并放入容器中。

如果该类中的成员变量上使用了诸如@Autowired和@Resource注解时,容器将会找对应的bean并注入–依赖注入。

而在多线程实例中使用@Autowired注解有时得不到对象,为什么呢?

线程类实例如下:

@Component
public class TestThread extends Thread{
    private static final Logger log = LoggerFactory.getLogger(TestThread.class);
    @Autowired
    ISysBoundUserExtendDao boundUserExtendDao;
    @Autowired
    SysBoundUserMapper boundUserMapper;
    int pageBegin;
    int pageEnd;
    public TestThread( int pageEnd,int pageBegin) {
        super();
        this.pageEnd = pageEnd;
        this.pageBegin = pageBegin;
        log.debug("pageBegin : "+pageBegin+" , pageEnd : "+pageEnd);
    }
    public TestThread() {
        super();
    }
    public TestThread(ISysBoundUserExtendDao boundUserExtendDao, SysBoundUserMapper boundUserMapper, int pageEnd,
            int pageBegin) {
        super();
        this.boundUserExtendDao = boundUserExtendDao;
        this.boundUserMapper = boundUserMapper;
        this.pageEnd = pageEnd;
        this.pageBegin = pageBegin;
        log.debug("pageBegin : "+pageBegin+" , pageEnd : "+pageEnd);
    }
    public void run(){
        String result = "";
        Integer pageSize=1000;
        Map<String, Object> queryMap = new HashMap<>();
        for(int i=pageBegin;i<pageEnd;i++){
            queryMap.put("page", i);
            queryMap.put("pageSize", pageSize);
            log.debug("开始获取第"+i+"页用户...");
            List<SysBoundUser> boundUserList = boundUserExtendDao.getBoundUserList(queryMap);
            //...
            }
        }
}

使用了@Component和@Autowired注解。

此时创建多线程如下:

    @RequestMapping(value="checkUser2",produces="application/json;charset=utf-8")
    @ResponseBody
    public String checkUser2(){
        String result = "";
        TestThread testThread1 = new TestThread(4, 3);
        testThread1.start();
        TestThread testThread2 = new TestThread( 5, 4);
        testThread2.start();
        TestThread testThread3 = new TestThread(6, 5);
        testThread3.start();
        TestThread testThread4 = new TestThread(7, 6);
        testThread4.start();
        TestThread testThread5 = new TestThread(8, 7);
        testThread5.start();
        return result;
    }

会发现,在run()方法中boundUserExtendDao为null!!!

这是因为创建多线程时,是new的对象实例,非容器扫描时自动创建的对象实例!!!而你创建线程的时候并没有传入boundUserExtendDao。

如果获取的是容器创建的TestThread,那么肯定有mapper和dao:



那么如何在多线程中使用boundUserExtendDao呢?

答:创建的时候传入boundUserExtendDao实例!


实例如下,此时run方法中可以正确拿到mapper和dao:

@Controller
@RequestMapping("/test")
public class TestDataController {
    private static final Logger log = LoggerFactory.getLogger(TestDataController.class);
    @Autowired
    ISysBoundUserExtendDao boundUserExtendDao;
    @Autowired
    SysBoundUserMapper boundUserMapper;
    @RequestMapping(value="checkUser2",produces="application/json;charset=utf-8")
    @ResponseBody
    public String checkUser2(){
        String result = "";
        TestThread testThread1 = new TestThread( boundUserExtendDao, boundUserMapper,4, 3);
        testThread1.start();
        TestThread testThread2 = new TestThread( boundUserExtendDao, boundUserMapper,5, 4);
        testThread2.start();
        TestThread testThread3 = new TestThread(boundUserExtendDao, boundUserMapper, 6, 5);
        testThread3.start();
        TestThread testThread4 = new TestThread(boundUserExtendDao, boundUserMapper, 7, 6);
        testThread4.start();
        TestThread testThread5 = new TestThread(boundUserExtendDao, boundUserMapper, 8, 7);
        testThread5.start();
        return result;
    }


目录
相关文章
|
Java Spring
Spring在多线程中bean的注入问题
Spring在多线程中bean的注入问题
573 0
|
安全 Java
为什么直接通过 @Autowired 注入的 HttpServletRequest 没有线程安全的问题
我们在各个地方注入依赖时,大多数情况下都是单例的。为什么直接通过 @Autowired 注入的 HttpServletRequest 没有线程安全的问题呢?带着这个问题我做了如下笔记。
517 0
|
安全 API Go
我的免杀之路:远程线程注入
远程线程注入技术能实现在Windows系统下进程的隐藏。其主要核心在于一个Windows API函数CreateRemoteThread,通过它可以在另外一个进程中注入一个线程并执行。
1541 0
我的免杀之路:远程线程注入
|
Windows
突破SESSION 0 隔离的远线程注入
在Windows XP,Windows Server 2003以及更早的版本中,第一个登录的用户以及Windows的所有服务都运行在Session 0上,这样的做法导致用户使用的应用程序可能会利用Windows的服务程序提升自身的权限,为此,在后续的Windows版本中,引入了一种隔离机制,普通应用程序已经不再session 0中运行。
395 0
|
前端开发 安全 Java
Spring注入的成员属性HttpServletRequest是线程安全的吗?【享学Spring MVC】(上)
Spring注入的成员属性HttpServletRequest是线程安全的吗?【享学Spring MVC】(上)
|
XML 安全 前端开发
Spring注入的成员属性HttpServletRequest是线程安全的吗?【享学Spring MVC】(下)
Spring注入的成员属性HttpServletRequest是线程安全的吗?【享学Spring MVC】(下)
Spring注入的成员属性HttpServletRequest是线程安全的吗?【享学Spring MVC】(下)
|
Java Linux Android开发
【Android 逆向】Android 进程注入工具开发 ( 远程进程注入动态库文件操作 | 注入动态库 加载 业务动态库 | 业务动态库启动 | pthread_create 线程开发 )
【Android 逆向】Android 进程注入工具开发 ( 远程进程注入动态库文件操作 | 注入动态库 加载 业务动态库 | 业务动态库启动 | pthread_create 线程开发 )
347 0
|
SQL 前端开发 安全
SpringBoot项目使用多线程处理任务时无法通过Autowired注入bean
  最近在做一个“温湿度控制”的项目,项目要求通过用户设定的温湿度数值和实时采集到的数值进行比对分析,因为数据的对比与分析是一个通过前端页面控制的定时任务,经理要求在用户开启定时任务时,单独开启一个线程进行数据的对比分析,并将采集到的温湿度数值存入数据库中的历史数据表,按照我们正常的逻辑应该是用户在请求开启定时任务时,前端页面通过调用后端接口,创建一个新的线程来执行定时任务,然后在线程类中使用 @Autowired 注解注入保存历史数据的service层,在线程类中调用service层保存历史数据的方法实现温湿度数据的保存,这时就出现了一个很尴尬的问题,在新开启的线程中使用 @Autowire
509 0

热门文章

最新文章