如上两图所示,在WebSocket中我想使用Redis。把自己编写的RedisUtil使用@Autowired自动注入到当前类。

在运行时,出现异常:java.lang.NullPointException (上面第二张图的代码)

A.可能原因:自己编写的RedisUtil没有放到spring容器中(导致没有实例化)。经过测试,排除此原因。(测试过程略)

B.解决思路:WebSocket是线程安全的,有用户连接时就会创建一个新的端点实例,一个端点只能保证一个线程调用。总结就是,WebSocket是多对象的。不管单独使用也好,结合spring也好,或者结合SpringBoot也罢,他都是多对象的。

C.问题原因:WebSocket是多对象的,使用的spring却是单例模式。这两者刚好冲突。@Autowired注解注入对象是在启动的时候就把对象注入,而不是在使用A对象时才把A需要的B对象注入到A中。而WebSocket在刚刚有说到,有连接时才实例化对象,而且有多个连接就有多个对象。由此得知,RedisUtil根本就没有注入到WebSocket当中。

D.解决问题:解决的方式有两种。1.让Redis属于WebSocketServer这个类   2.在运行时,动态的从spring容器中取出RedisUtil。

第一种:

 private static RedisUtil redisUtil;
 
    @Autowired
    public static void setRedisUtil(RedisUtil redisUtil) {
        WebSocketServer.redisUtil = redisUtil;
    }
第二种:         

编写从容器中取对象的工具类

@Component
public class SpringUtil implements ApplicationContextAware {
    private static ApplicationContext applicationContext;
 
    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        SpringUtil.applicationContext = applicationContext;
    }
 
    public ApplicationContext getApplicationContext(){
        return applicationContext;
    }
 
    public static Object getBean(String beanName){
        return applicationContext.getBean(beanName);
    }
 
    public static <T> T getBean(Class<T> clazz){
        return (T)applicationContext.getBean(clazz);
    }
}
 然后再WebSocket中

private RedisUtil redisUtil = SpringUtil.getBean(RedisUtil.class);
啊~行了行了,就到这了。

最后再哔哔一句,原理枯燥恶心。但解决问题的时候真香!~
————————————————
版权声明:本文为CSDN博主「王百逸」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/Programmer__Wang/article/details/88538993

Logo

权威|前沿|技术|干货|国内首个API全生命周期开发者社区

更多推荐