原因分析

在多线程时使用@Autowired总是获取不到bean,原因是:new thread不在spring容器中,也就无法获得spring中的bean对象。

解决办法

手动获取,新建一个工具类,代码如下

package com.gdie.modules.oplog.util;

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

/**
 * 解决多线程处理任务时无法通过@Autowired注入bean
 */
@Component
public class ApplicationContextProvider implements ApplicationContextAware {
	/**
	 * 上下文对象实例
	 */
	private static ApplicationContext applicationContext;

	@Override
	public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
		this.applicationContext = applicationContext;
	}

	/**
	 * 获取applicationContext
	 *
	 * @return
	 */
	public static ApplicationContext getApplicationContext() {
		return applicationContext;
	}

	/**
	 * 通过name获取 Bean.
	 *
	 * @param name
	 * @return
	 */
	public static Object getBean(String name) {
		return getApplicationContext().getBean(name);
	}

	/**
	 * 通过class获取Bean.
	 *
	 * @param clazz
	 * @param <T>
	 * @return
	 */
	public static <T> T getBean(Class<T> clazz) {
		return getApplicationContext().getBean(clazz);
	}

	/**
	 * 通过name,以及Clazz返回指定的Bean
	 *
	 * @param name
	 * @param clazz
	 * @param <T>
	 * @return
	 */
	public static <T> T getBean(String name, Class<T> clazz) {
		return getApplicationContext().getBean(name, clazz);
	}
}

然后线程类中写一个无参的构造方法,在构造方法中,通过调用工具类中的 getBean() 方法就可以拿到实例了,程序在调用这个线程类时,会自动调用其无参的构造方法,在构造方法中我们将需要的bean对象注入,然后就可以正常使用了。

import org.springframework.beans.factory.annotation.Autowired;

import com.gdie.modules.oplog.service.LogDBOutput;
import com.gdie.modules.oplog.util.ApplicationContextProvider;

/**
 * 保存日志的线程
 * @Description :
 * @date : 2022年7月27日
 */
public class LogInfo implements Runnable {

    @Autowired
    private LogDBOutput logDBOutput ApplicationContextProvider.getBean(LogDBOutput.class);
	

    @Override
    public void run() {
        // 这里就可以调用LogDBOutput中的方法了
        logDBOutput.addLog();
    }