转自https://blog.csdn.net/Programmer__Wang/article/details/88538993
参考https://blog.csdn.net/kxj19980524/article/details/85108047
使用service
//本质原因:spring管理的都是单例(singleton)和 websocket (多对象)相冲突。
public static DeviceService deviceService;
@Autowired
public void setDeviceService(DeviceService deviceService){
WebSocketController.deviceService = deviceService;
}
如上两图所示,在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