问题出现的原因
在使用ThreadLocal存储用户的openid时(这里的openid是用于微信小程序开发的,在你的开发环境中可以换成任意的你想存放在ThreadLocal即本地线程中的变量),需要注意ThreadLocal的作用范围。ThreadLocal是基于线程的,它为每个线程提供了一个独立的副本,并且只能在同一个线程内获取到存储的值。
在WebSocket的生命周期中,afterConnectionEstablished和afterConnectionClosed方法运行在不同的线程上。在afterConnectionEstablished方法中,WebSocket连接刚建立时,该方法是在与客户端进行握手的线程上执行的,而在afterConnectionClosed方法中,WebSocket连接已关闭,该方法是在释放连接资源的线程上执行的。
因此,如果在拦截器中将openid存储在ThreadLocal中,而在afterConnectionClosed方法中尝试获取该值,可能由于线程不同而无法获取到正确的值。
解决方案
解决这个问题的一种常见方法是,将用户的openid存储到WebSocketSession的属性中,在afterConnectionEstablished方法中设置属性,在afterConnectionClosed方法中获取属性。WebSocketSession是在整个WebSocket连接期间持久存在的对象,可以跨不同的线程访问。