我的编程空间,编程开发者的网络收藏夹
学习永远不晚

如何解决Spring对静态变量无法注入的问题

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

如何解决Spring对静态变量无法注入的问题

这篇文章主要讲解了“如何解决Spring对静态变量无法注入的问题”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“如何解决Spring对静态变量无法注入的问题”吧!

Spring对静态变量无法注入

问题

今天在学习的过程中想写一个连接和线程绑定的JDBCUtils工具类,但测试时发现一直报空指针异常,上网查了之后Spring并不支持对静态成员变量注入,所以光试用@Autowired肯定是不行的。

可是我们编写工具类时肯定是要使用静态变量和方法的,我总结一下我用过可以实现对静态成员变量注入的方法。

@Componentpublic class JDBCUtils {    @Autowired    private static ComboPooledDataSource dataSource;    private static ThreadLocal<Connection> tl = new ThreadLocal<Connection>();    public static Connection getThreadConnection(){        Connection conn = tl.get();        if (conn == null){            conn = getConnection();            tl.set(conn);        }        return conn;    }    public static DataSource getDataSource(){        return dataSource;    }    public static Connection getConnection(){        Connection connection = null;        try {            connection = dataSource.getConnection();        } catch (SQLException e) {            e.printStackTrace();        }        return connection;    }    public static void removeThreadConnection(){        tl.remove();    }}

set方法注入

注解方式

在类前加@Component注解,在set方法上加 @Autowired注解,这里注意两点

配置文件里已经配置了变量的相关参数

静态变量自动生成set方法时会有static修饰,要去掉,否则还是无法注入

@Componentpublic class JDBCUtils {    private static ComboPooledDataSource dataSource;    @Autowired    public void setDataSource(ComboPooledDataSource dataSource) {        JDBCUtils.dataSource = dataSource;    }
xml方式

同样注意将set方法上的static去掉

public class JDBCUtils {    private static   ComboPooledDataSource dataSource;    public void setDataSource(ComboPooledDataSource dataSource) {        this.dataSource = dataSource;    }    private static ThreadLocal<Connection> tl = new ThreadLocal<Connection>();    public static Connection getThreadConnection(){        Connection conn = tl.get();        if (conn == null){            conn = getConnection();            tl.set(conn);        }        return conn;    }    public static DataSource getDataSource(){        return dataSource;    }    public static Connection getConnection(){        Connection connection = null;        try {            connection = dataSource.getConnection();        } catch (SQLException e) {            e.printStackTrace();        }        return connection;    }    public static void removeThreadConnection(){        tl.remove();    }}
   <bean id="JDBCUtils" class="com.cc.utils.JDBCUtils">        <property name="dataSource" ref="dataSource"></property>    </bean>

@PostConstruct注解方式注入

用@PostConstruct加在init方法上,在类初始化后执行该方法,对成员变量赋值。在这之前,我们要改造一下工具类,去掉我们想注入变量的static的修饰符,这样我们就可以用@Autowired实现对其注入。

然后加一个静态的类自身的引用对象,当我们想要变量时通过这个引用对象来获取。

@Componentpublic class JDBCUtils {    @Autowired    private  ComboPooledDataSource dataSource;    private static JDBCUtils jdbcUtils;    @PostConstruct    public void init(){        jdbcUtils = this;        this.dataSource = dataSource;    }    private static ThreadLocal<Connection> tl = new ThreadLocal<Connection>();    public static Connection getThreadConnection(){        Connection conn = tl.get();        if (conn == null){            conn = getConnection();            tl.set(conn);        }        return conn;    }    public static DataSource getDataSource(){        return jdbcUtils.dataSource;    }    public static Connection getConnection(){        Connection connection = null;        try {            connection = jdbcUtils.dataSource.getConnection();        } catch (SQLException e) {            e.printStackTrace();        }        return connection;    }    public static void removeThreadConnection(){        tl.remove();    }}

当然这种用初始化方法也可以用xml配置,原理一样。

public class JDBCUtils {    private  ComboPooledDataSource dataSource;    public void setDataSource(ComboPooledDataSource dataSource) {        this.dataSource = dataSource;    }    private static JDBCUtils jdbcUtils;    public void init(){        jdbcUtils = this;        this.dataSource = dataSource;    }    private static ThreadLocal<Connection> tl = new ThreadLocal<Connection>();    public static Connection getThreadConnection(){        Connection conn = tl.get();        if (conn == null){            conn = getConnection();            tl.set(conn);        }        return conn;    }    public static DataSource getDataSource(){        return jdbcUtils.dataSource;    }    public static Connection getConnection(){        Connection connection = null;        try {            connection = jdbcUtils.dataSource.getConnection();        } catch (SQLException e) {            e.printStackTrace();        }        return connection;    }    public static void removeThreadConnection(){        tl.remove();    }}
<bean id="JDBCUtils" class="com.cc.utils.JDBCUtils" init-method="init">        <property name="dataSource" ref="dataSource"></property>    </bean>

静态方法注入bean失败原因

今天在写redission 的一个工具类的时候,随手写出下面的代码

package com.wt.redission.wtredission.utils;  import org.redisson.api.*;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.stereotype.Component; import javax.annotation.PostConstruct; @Componentpublic class RedissionUtilserror {     @Autowired   private static RedissonClient redissonClient;      public static RLock getRLock(String objectName) {        RLock rLock =redissonClient.getLock(objectName);        return rLock;    }      //根据名字获取map    public static  <K, V> RMap<K, V> getRMap(String objectName) {        RMap<K, V> map = redissonClient.getMap(objectName);        return map;    }     //根据名字和值设置map    public static void setMap(String objectName,Object key,Object value){        RMap<Object, Object> map =redissonClient.getMap(objectName);        map.put(key,value);    }      //根据名字获取set    public static <V> RSet<V> getSet(String objectName) {        RSet<V> set = redissonClient.getSet(objectName);        return set;    }     //根据名字和值设置set    public static void setSet(String objectName,Object value){        RSet<Object> set = redissonClient.getSet(objectName);        set.add(value);    }     //根据名字获取list    public static  <V> RList<V> getRList(String objectName) {        RList<V> rList = redissonClient.getList(objectName);        return rList;    }      //根据名字和值设置list    public static void setList(String objectName, int  index,Object element ){        RList<Object> objectRList = redissonClient.getList(objectName);        objectRList.set(index,element);    }     //根据名字获取bucket    public static <T> RBucket<T> getRBucket(String objectName) {        RBucket<T> bucket = redissonClient.getBucket(objectName);        return bucket;    }     //根据名字和值 设置对应的bucket    public static  <T> T setBucket(String objectName,String value){        RBucket<Object> bucket = redissonClient.getBucket(objectName);        bucket.set(value);        T t= (T) bucket.get(); //值类型由返回值确定        return  t;    } }

乍一看好像没问题 我写一个静态方法 然后在方法中使用静态变量redissonClient ,哇....,一切看得如此正常

当我开始测试时,NPE.............,我去这是怎么回事,自己在想这不科学啊,怎么会空指针,于是我开始找原因

最后发现是基础不牢啊............,对jvm的类加载机制几乎就没考虑,简要说要错误的原因

jvm在进行类加载的时候,首先会加载类变量,类方法,也就是我这里被static修饰的方法,然后当我调用静态方法进行使用的时候,会使用到redissionClient,注意这个redissionClient是通过autowired进来的,关键问题就在这里,autowired的底层是通过构造器和set方法注入bean的

redissionClient被static修饰 并且还是一个接口 在被调用的时候肯定没有实例化

下面提供三种方式正确使用

方式一

package com.wt.redission.wtredission.utils;  import org.redisson.api.*;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.stereotype.Component; import java.util.List; @Componentpublic class RedissionUtils {     private static RedissonClient redissonClient;     @Autowired    public  RedissionUtils(RedissonClient redissonClient){        RedissionUtils.redissonClient=redissonClient;    }       public static RLock getRLock(String objectName) {        RLock rLock = redissonClient.getLock(objectName);        return rLock;    }      //根据名字获取map    public static  <K, V> RMap<K, V> getRMap(String objectName) {        RMap<K, V> map = redissonClient.getMap(objectName);        return map;    }     //根据名字和值设置map    public static void setMap(String objectName,Object key,Object value){        RMap<Object, Object> map =redissonClient.getMap(objectName);        map.put(key,value);    }      //根据名字获取set    public static <V> RSet<V> getSet(String objectName) {        RSet<V> set = redissonClient.getSet(objectName);        return set;    }     //根据名字和值设置set    public static void setSet(String objectName,Object value){        RSet<Object> set = redissonClient.getSet(objectName);        set.add(value);    }     //根据名字获取list    public static  <V> RList<V> getRList(String objectName) {        RList<V> rList = redissonClient.getList(objectName);        return rList;    }      //根据名字和值设置list    public static void setList(String objectName, int  index,Object element ){        RList<Object> objectRList = redissonClient.getList(objectName);        objectRList.set(index,element);    }     //根据名字获取bucket    public static <T> RBucket<T> getRBucket(String objectName) {        RBucket<T> bucket = redissonClient.getBucket(objectName);        return bucket;    }     //根据名字和值 设置对应的bucket    public static  <T> T setBucket(String objectName,String value){        RBucket<Object> bucket = redissonClient.getBucket(objectName);        bucket.set(value);        T t= (T) bucket.get(); //值类型由返回值确定        return  t;    } }

方式二

package com.wt.redission.wtredission.utils;  import org.redisson.api.*;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.stereotype.Component; import javax.annotation.PostConstruct; @Componentpublic class RedissionUtils2 {     @Autowired    RedissonClient redissonClient;     public  static RedissionUtils2 redissionUtils;     @PostConstruct    public  void  init(){        redissionUtils=this;        redissionUtils.redissonClient=this.redissonClient;    }      public static RLock getRLock(String objectName) {        RLock rLock = redissionUtils.redissonClient.getLock(objectName);        return rLock;    }      //根据名字获取map    public static  <K, V> RMap<K, V> getRMap(String objectName) {        RMap<K, V> map = redissionUtils.redissonClient.getMap(objectName);        return map;    }     //根据名字和值设置map    public static void setMap(String objectName,Object key,Object value){        RMap<Object, Object> map =redissionUtils.redissonClient.getMap(objectName);        map.put(key,value);    }      //根据名字获取set    public static <V> RSet<V> getSet(String objectName) {        RSet<V> set = redissionUtils.redissonClient.getSet(objectName);        return set;    }     //根据名字和值设置set    public static void setSet(String objectName,Object value){        RSet<Object> set = redissionUtils.redissonClient.getSet(objectName);        set.add(value);    }     //根据名字获取list    public static  <V> RList<V> getRList(String objectName) {        RList<V> rList = redissionUtils.redissonClient.getList(objectName);        return rList;    }      //根据名字和值设置list    public static void setList(String objectName, int  index,Object element ){        RList<Object> objectRList = redissionUtils.redissonClient.getList(objectName);        objectRList.set(index,element);    }     //根据名字获取bucket    public static <T> RBucket<T> getRBucket(String objectName) {        RBucket<T> bucket = redissionUtils.redissonClient.getBucket(objectName);        return bucket;    }     //根据名字和值 设置对应的bucket    public static  <T> T setBucket(String objectName,String value){        RBucket<Object> bucket = redissionUtils.redissonClient.getBucket(objectName);        bucket.set(value);        T t= (T) bucket.get(); //值类型由返回值确定        return  t;    } }

方式三 通过spring上下文获取

package com.wt.redission.wtredission.utils;  import io.micrometer.core.instrument.util.StringUtils;import org.springframework.beans.BeansException;import org.springframework.beans.factory.NoSuchBeanDefinitionException;import org.springframework.context.ApplicationContext;import org.springframework.context.ApplicationContextAware;import org.springframework.context.annotation.Scope;import org.springframework.stereotype.Component; import javax.servlet.http.HttpServletRequest; @Component@Scope("singleton")public class SpringUtil implements ApplicationContextAware {         private static ApplicationContext applicationContext;         @Override    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {SpringUtil.applicationContext = applicationContext;    }         public static ApplicationContext getApplicationContext() {        return applicationContext;    }         @SuppressWarnings("unchecked")    public static <T> T getBean(String name) throws BeansException {        return (T) applicationContext.getBean(name);    }         public static <T> T getBean(Class<T> clz) throws BeansException {        return (T)applicationContext.getBean(clz);    }         public static boolean containsBean(String name) {        return applicationContext.containsBean(name);    }         public static boolean isSingleton(String name) throws NoSuchBeanDefinitionException {        return applicationContext.isSingleton(name);    }         public static Class<?> getType(String name) throws NoSuchBeanDefinitionException {        return applicationContext.getType(name);    }         public static String[] getAliases(String name) throws NoSuchBeanDefinitionException {        return applicationContext.getAliases(name);    }  public static String getJwtToken(HttpServletRequest servletRequest, String tokenId) {String token = servletRequest.getHeader(tokenId);if (StringUtils.isBlank(token)) {token = servletRequest.getParameter(tokenId);}return token;}}

感谢各位的阅读,以上就是“如何解决Spring对静态变量无法注入的问题”的内容了,经过本文的学习后,相信大家对如何解决Spring对静态变量无法注入的问题这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是编程网,小编将为大家推送更多相关知识点的文章,欢迎关注!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

如何解决Spring对静态变量无法注入的问题

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

如何解决Spring对静态变量无法注入的问题

这篇文章主要讲解了“如何解决Spring对静态变量无法注入的问题”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“如何解决Spring对静态变量无法注入的问题”吧!Spring对静态变量无法注入
2023-06-20

如何解决Spring自动注入失败的问题

小编给大家分享一下如何解决Spring自动注入失败的问题,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!Spring自动注入失败如何解决?我有一个被Spring @
2023-06-15

如何解决@Table注解无法使用的问题

这篇文章主要介绍了如何解决@Table注解无法使用的问题,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。@Table注解无法使用及报红在新建实体类时想直接序列化,遇到注解不起作
2023-06-26

如何解决SpringBoot引用别的模块无法注入的问题

小编给大家分享一下如何解决SpringBoot引用别的模块无法注入的问题,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!引用别的模块无法注入的问题前段时间尝试着利用
2023-06-29

Linux service无法使用系统环境变量问题如何解决

这篇“Linux service无法使用系统环境变量问题如何解决”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Linux s
2023-07-05

如何解决springMVC跳转js、css图片等静态资源无法加载的问题

这篇文章将为大家详细讲解有关如何解决springMVC跳转js、css图片等静态资源无法加载的问题,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。web.xml中servlet>
2023-05-31

spring注解在自定义jar包中无法被扫描的问题如何解决

这篇文章主要介绍“spring注解在自定义jar包中无法被扫描的问题如何解决”,在日常操作中,相信很多人在spring注解在自定义jar包中无法被扫描的问题如何解决问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答
2023-06-20

如何解决win8无法进入系统的问题

这篇文章主要介绍了如何解决win8无法进入系统的问题,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。原因一、外接设备导致解决方法:一般来说,遇到这种问题,可能是外接设备导致,我
2023-06-28

SpringBoot+Spring Security无法实现跨域的问题如何解决

本篇内容主要讲解“SpringBoot+Spring Security无法实现跨域的问题如何解决”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“SpringBoot+Spring Security
2023-06-20

如何解决Mac无法写入U盘的问题

这期内容当中小编将会给大家带来有关如何解决Mac无法写入U盘的问题,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。为了让Mac支持读写ntfs硬盘,许多人都会花钱购买第三方软件。这类软件不仅价格高,还相当不
2023-06-05

如何解决python面对用户无意义输入的问题

这篇文章主要介绍如何解决python面对用户无意义输入的问题,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!问题正在编写一个接受用户输入的程序。#note: Python 2.7 users should use `r
2023-06-20

如何解决虚拟机无法导出导入的问题

检查虚拟机是否受损:首先检查虚拟机是否受损或出现错误。可以尝试重新启动虚拟机或使用虚拟机管理软件进行修复。检查虚拟机配置:确保导出和导入的虚拟机配置是兼容的。如果导出的虚拟机配置与导入的虚拟机配置不匹配,可能会导致导入失败。使用正确的导出和
如何解决虚拟机无法导出导入的问题
2024-06-10

如何解决Callable的对象中用@Autowired注入别的对象失败问题

这篇文章给大家分享的是有关如何解决Callable的对象中用@Autowired注入别的对象失败问题的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。实现Callable的对象中,用@Autowired注入别的对象失
2023-06-20

Ubuntu12.04下如何解决无法使用中文输入法的问题

这篇文章主要介绍“Ubuntu12.04下如何解决无法使用中文输入法的问题”,在日常操作中,相信很多人在Ubuntu12.04下如何解决无法使用中文输入法的问题问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”U
2023-06-13

如何解决mybatis-generator生成器添加类注释方法无效的问题

本篇内容主要讲解“如何解决mybatis-generator生成器添加类注释方法无效的问题”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“如何解决mybatis-generator生成器添加类注释
2023-06-20

如何解决笔记本电脑安装flash提示错误无法注册的问题

这篇文章给大家分享的是有关如何解决笔记本电脑安装flash提示错误无法注册的问题的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。原因分析:1、用户权限不够,不拥有Flash Player 注册表项写入权限。2、之前
2023-06-27

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录