线程中使用@Autowired注入

线程中使用@Autowired注入

场景

多多线程同时访问接口,测试接口稳定性

工具RestTemplate

使用springboot自带http调用
线程类实现Callable

代码实现

一下是通过@Autowired注入的方式

public static void main(String[] arge){
            Future p = pool.submit(new RunnableUtil(url, urlSuffixInsertNews, body1, tokenDto.getToken()));
            JSONObject resJson = JSONObject.parseObject(p.get().toString());
}

线程类

package com.yx.util.runnable;

import com.alibaba.fastjson.JSONObject;
import com.yx.util.httpClient.ClientUtil;
import lombok.Data;
import java.util.Map;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

@Data
public class RunnableUtil implements Callable {

	@Autowired
    private ClientUtil clientUtil;
    
    private String url;
    private Map<String, Object> json;
    private String urlSuffix;
    private String token;
    public RunnableUtil(String url, String urlSuffix, Map<String, Object> json, String token){
        super();
        this.url = url;
        this.urlSuffix = urlSuffix;
        this.json = json;
        this.token = token;
    }
    @Override
    public Object call() throws Exception {
        String res = clientUtil.doPost(url, urlSuffix, json, token);
        return res;
    }
}

请求方法

package com.yx.util.httpClient;


import org.springframework.http.HttpEntity;
import org.springframework.http.HttpHeaders;
import org.springframework.http.ResponseEntity;
import org.springframework.web.client.RestTemplate;
import org.springframework.web.util.DefaultUriBuilderFactory;

import java.util.Map;
@Component
public class ClientUtil {

	 @Autowired
    private RestTemplate restTemplate;
    /**
     * @param url ip+端口 :127.0.0.1:8080/
     * @param urlSuffix 请求地址
     * @param body 请求参数 json格式
     * @param token 请求头参数
     * */
    public static String doPost(String url, String urlSuffix, Map<String, Object> body, String token){

        DefaultUriBuilderFactory factory = new DefaultUriBuilderFactory(url);
        factory.setEncodingMode(DefaultUriBuilderFactory.EncodingMode.TEMPLATE_AND_VALUES);
        restTemplate.setUriTemplateHandler(factory);
        HttpHeaders headers = new HttpHeaders();
        headers.add("token", token);

        HttpEntity<Map> requestEntity = new HttpEntity<>(body, headers);
        ResponseEntity<String> responseEntity = restTemplate.postForEntity(urlSuffix,requestEntity , String.class);
        if(responseEntity.getStatusCode().is2xxSuccessful()){
            System.out.println(responseEntity.getBody());
        }else {
            System.out.println(responseEntity.getStatusCode().value()+ "和" + responseEntity.getBody());
        }
        return responseEntity.getBody();
    }
}

也就是线程@通过Autowired注入的方式调用会一直出现空指针异常
解决方案就是在晚上找一个获取bean的方法

@Component
public class SpringContextUtil implements ApplicationContextAware {
    /**
     * 上下文对象实例
     */
    private static ApplicationContext applicationContext;
 
    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        SpringContextUtil.applicationContext = applicationContext;
    }
 
    /**
     * 获取applicationContext
     *
     * @return
     */
    public static ApplicationContext getApplicationContext() {
        return applicationContext;
    }
 
    /**
     * 通过name获取 Bean.
     *
     * @param name
     * @return
     */
    public static Object getBean(String name) {
        if (applicationContext == null){
            throw new RuntimeException("applicationContext注入失败");
        }
        return getApplicationContext().getBean(name);
    }
 
    /**
     * 通过class获取Bean.
     *
     * @param clazz
     * @param <T>
     * @return
     */
    public static <T> T getBean(Class<T> clazz) {
        if (applicationContext == null){
            throw new RuntimeException("applicationContext注入失败");
        }
        return getApplicationContext().getBean(clazz);
    }
 
    /**
     * 通过name,以及Clazz返回指定的Bean
     *
     * @param name
     * @param clazz
     * @param <T>
     * @return
     */
    public static <T> T getBean(String name, Class<T> clazz) {
        if (applicationContext == null){
            throw new RuntimeException("applicationContext注入失败");
        }
        return getApplicationContext().getBean(name, clazz);
    }
}

使用

RestTemplate restTemplate =  SpringContextUtil.getBean(ClientUtil.class)
  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
多线程环境下,使用@Autowired注解时可能会出现失效的情况。原因是多线程环境下,多个线程同时访问同一个对象的实例,可能出现竞争条件,导致注入失败。 自动注入(@Autowired)是Spring框架提供的依赖注入的方式,用于自动装配Bean对象。当多个线程同时使用@Autowired注解来注入同一个对象时,可能会出现以下情况之一: 1. 线程之间存在竞争条件:如果多个线程同时对同一个对象进行注入,可能会导致其一个线程注入被后来的线程覆盖,最终只有最后一个线程成功注入。 2. Bean对象的状态改变:在多线程环境下,如果多个线程同时对Bean对象进行修改,可能会导致Bean对象的状态发生变化。而@Autowired注解是在Bean对象创建完成后进行注入的,如果Bean对象的状态改变了,可能会导致注入失败。 为了解决多线程环境下@Autowired失效的问题,可以考虑以下方案: 1. 使用局部变量:在多线程环境下,将需要注入的对象作为局部变量使用,而不是直接注入到类成员变量。每个线程使用独立的局部变量,可以避免线程之间的竞争条件。 2. 使用线程安全的依赖注入方式:可以使用线程安全的依赖注入方式,如构造函数注入或Setter方法注入。这些方式可以保证在多线程环境下对象的正确注入。 3. 使用ThreadLocal:可以使用ThreadLocal来保证每个线程都有独立的对象实例。在需要使用Autowired注入的地方,通过ThreadLocal获取对象实例,确保每个线程使用的是独立的实例。 总之,多线程环境下使用@Autowired注解时可能会出现失效的情况,需要针对具体的情况选择合适的解决方案来避免注入失效问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值