SpringBoot使用@ServerEndpoint无法依赖注入问题解决(WebSocket)

如上两图所示,在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

以下是一个可以通过 WebSocket 传输图片,视频和文档的示例代码,其中 `@ServerEndpoint` 注解用于创建 WebSocket 服务器端点: ```java import java.io.IOException; import java.nio.ByteBuffer; import java.util.Base64; import javax.websocket.OnMessage; import javax.websocket.Session; import javax.websocket.server.ServerEndpoint; @ServerEndpoint("/file-transfer") public class FileTransferEndpoint { @OnMessage public void onMessage(Session session, ByteBuffer buffer) throws IOException { byte[] data = buffer.array(); String type = session.getUserProperties().getOrDefault("type", "").toString(); if (type.equals("image")) { String base64 = Base64.getEncoder().encodeToString(data); session.getBasicRemote().sendText("data:image/png;base64," + base64); } else if (type.equals("video")) { session.getBasicRemote().sendBinary(buffer); } else { session.getBasicRemote().sendBinary(buffer); } } } ``` 上述代码中,`onMessage` 方法用于处理客户端发送的消息,其中 `ByteBuffer` 对象可以用于传输二进制数据。通过 `Session` 对象的 `getUserProperties` 方法可以获取客户端发送的文件类型(如图片、视频或文档),并根据不同类型的文件进行处理。如果收到的是图片,将二进制数据转换为 Base64 编码的字符串,并添加图片的数据类型后发送;如果收到的是视频或文档,直接将二进制数据发送回客户端。 在客户端发送文件时,可以使用 JavaScript 的 `FileReader` 对象将文件读取为二进制数据,并通过 WebSocket 发送到服务器端: ```javascript var file = document.getElementById("file").files[0]; var reader = new FileReader(); reader.onload = function(event) { var data = event.target.result; var buffer = new Uint8Array(data); socket.send(buffer); }; reader.readAsArrayBuffer(file); ``` 上述代码中,`file` 对象表示要发送的文件,`FileReader` 对象用于将文件读取为二进制数据,`Uint8Array` 对象用于将二进制数据转换为 `ArrayBuffer` 对象,最后调用 WebSocket 的 `send` 方法发送数据到服务器端。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值