小坑:使用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");
}
}
}然后在服务端这样配置:
@serverendpoint(value ="/chatroom/{username}",configurator=httpsessionwshelper.class,encoders = {serverencoder.class})
//encoders = {serverencoder.class}用于指定sendobject()方法调用的解析器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
}
}网上解释说,是因为进行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");然而浏览器地址栏是这样的:
http://localhost:8080/
网上解释说如果不使用同一个host,则会创建不同的连接请求。具体细节有兴趣的朋友可以去了解一下。
于是我们可以这样拼接一下:
var host = window.location.host; var url = "ws://"+host+"/chatroom/null";
这样便可以正常建立ws请求,并且能够使用自定义的httpsessionwshelper 类获取到httpsession了。
到此这篇关于websocket获取httpsession空指针异常的解决办法的文章就介绍到这了,更多相关websocket获取httpsession空指针内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!
发表评论