WebSocket获取httpSession空指针异常的解决办法


小坑:使用requestListner解决不了这个问题!

如何获取HttpSession

在使用webSocket实现p2p或者一对多聊天功能的时候我们经常会有这样的需求:webSocket服务端需要获取到用户使用数据库的用户信息登录后的HttpSession获取个人资料信息。
于是,你会使用这样的代码:

package com.xinyulee.ws; import javax.servlet.http.HttpSession; import javax.websocket.HandshakeResponse; import javax.websocket.server.HandshakeRequest; import javax.websocket.server.ServerEndpointConfig; import javax.websocket.server.ServerEndpointConfig.Configurator; /** * Created by zipple on 2017/11/14. * 协助server获取http session */ public class HttpSessionWSHelper extends Configurator { @Override public void modifyHandshake(ServerEndpointConfig sec, HandshakeRequest request, HandshakeResponse response) { System.out.println("调用modifyHandshake方法..."); HttpSession session = (HttpSession) request.getHttpSession();//session有可能为空 if (session!=null){ System.out.println("获取到session id:"+session.getId()); sec.getUserProperties().put(HttpSession.class.getName(),session); }else{ System.out.println("modifyHandshake 获取到null session"); } } } 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26

然后在服务端这样配置:

@ServerEndpoint(value ="/chatRoom/{username}",configurator=HttpSessionWSHelper.class,encoders = {ServerEncoder.class}) //encoders = {ServerEncoder.class}用于指定sendObject()方法调用的解析器 
  • 1
  • 2
  • 3

RequestListner解决不了这个问题的原因

在使用上述方法配置完成以后我满怀信心的进行测试,但是意外的发现了Tomcat Localhost Log下报了null pointer exception。
经过调试,发现了modifyHandshake方法并不能获取到HttpSession,在上述HttpSessionWSHelper 类代码中可以看到我对它进行了判空处理。但是这样并不能起到什么实际上的作用。我们需要弄明白为什么HandshakeRequest 获取不到HttpSession。
一开始,我试着去百度了一下,发现有这样的代码:

package com.xinyulee.listener; import com.xinyulee.util.Log; import javax.servlet.ServletRequestEvent; import javax.servlet.ServletRequestListener; import javax.servlet.annotation.WebListener; import javax.servlet.http.HttpServletRequest; /** * Created by zipple on 2017/11/14. */ @WebListener public class RequestListener implements ServletRequestListener { public void requestInitialized(ServletRequestEvent sre) { //将所有request请求都携带上httpSession----这样会创建新的session!导致重新创建了会话 Log.info("request--:"+sre.getServletRequest().getLocalAddr()); ((HttpServletRequest) sre.getServletRequest()).getSession(); } public RequestListener() { // TODO Auto-generated constructor stub } public void requestDestroyed(ServletRequestEvent arg0) { // TODO Auto-generated method stub } }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28

网上解释说,是因为进行ws连接的时候并没有HttpSession处于激活状态,于是便不能获取到HttpSession。他们给出的解决办法是对每一个request请求调用sre.getServletRequest()).getSession()方法。
这样的确可以解决空指针异常的问题,但是新问题又出现了。
在没有HttpSession激活状态的时候,使用getSession()方法会新建一个HttpSession,也就是说实际上这个监听器是在没有激活HttpSession的情况下不断新建会话。
这样已经完全偏离了我们程序的需求:获取同时在线的HttpSession中保存的个人信息。
那么怎么办呢?
事实上,这个空指针异常的根源并不是出在我们后端代码上。

localhost和127.0.0.1其实并不是同一个连接

在前端连接WebSocket的时候,我的代码是这样的:

   loadWS("ws://127.0.0.1/chatRoom/null");
  • 1

然而浏览器地址栏是这样的:

    http://localhost:8080/
  • 1

网上解释说如果不使用同一个host,则会创建不同的连接请求。具体细节有兴趣的朋友可以去了解一下。
于是我们可以这样拼接一下:

    var host = window.location.host; var url = "ws://"+host+"/chatRoom/null";
  • 1
  • 2

这样便可以正常建立ws请求,并且能够使用自定义的HttpSessionWSHelper 类获取到HttpSession了


免责声明!

本站转载的文章为个人学习借鉴使用,本站对版权不负任何法律责任。如果侵犯了您的隐私权益,请联系本站邮箱yoyou2525@163.com删除。



 
粤ICP备18138465号  © 2018-2025 CODEPRJ.COM