使用spring websocket實現(xiàn)后臺給前端推送消息
在支持websocket的瀏覽器版本上能夠正常實現(xiàn)消息的收發(fā),現(xiàn)在使用sockjs做為不支持websocket的瀏覽器的兼容方案。
瀏覽器支持websocket時:
并且后端會輸出對應的連接成功的日志信息
當瀏覽器不支持websocket時,使用sockjs:
發(fā)現(xiàn)握手請求發(fā)送成功了,但是并沒有握手成功,連接沒有建立起來,也沒有相關的日志信息輸出。
"/info"和"t="是sockjs自己加上去的,我原來的url是http://localhost:8081/FGMSP-ui/sockjs/websocket?uid=2131231
為什么sockjs會在我原有的url上自動加上info的后綴和t這個參數(shù)?
另外sockjs之后還發(fā)起過很多次類似的請求,如下圖
都是在我原先的url上加了后綴和參數(shù),這是什么原理???
為什么我的第一次的握手請求發(fā)送顯示成功了(在調試DispatcherServlet類的doDispatch()時也看到了這個請求已經進來),但是沒有被interceptor和handler處理?是我的配置出錯了嗎?
下面是我的一些代碼和配置
config
@Configuration
@EnableWebMvc
@EnableWebSocket
public class MySocketConfig extends WebMvcConfigurerAdapter implements WebSocketConfigurer {
@Autowired
private MySocketHandler mySocketHandler;
@Autowired
private MyInterceptor myInterceptor;
@Override
public void registerWebSocketHandlers(WebSocketHandlerRegistry registry) {
registry.addHandler(mySocketHandler, "/websocket")
.addInterceptors(myInterceptor);
// 瀏覽器不支持websocket,使用sockjs建立連接
registry.addHandler(new MySocketHandler(), "/sockjs/websocket/info")
.addInterceptors(myInterceptor)
.withSockJS();
}
}
interceptor
public class MyInterceptor implements HandshakeInterceptor {
@Override
public boolean beforeHandshake(ServerHttpRequest request, ServerHttpResponse response,
WebSocketHandler webSocketHandler, Map<String, Object> attributes) throws Exception {
// System.out.println("Websocket:用戶[ID:" + ((ServletServerHttpRequest) request).getServletRequest().getSession(false).getAttribute("uid") + "]已經建立連接");
if (request instanceof ServletServerHttpRequest) {
// 獲取請求參數(shù),首先我們要獲取HttpServletRequest對象才能獲取請求參數(shù);當ServerHttpRequset的層次結構打開后其子類可以獲取到我們想要的http對象,那么就簡單了。
// 我這里是把獲取的請求數(shù)據(jù)綁定到session的map對象中(attributes)
HttpServletRequest servletRequest = ((ServletServerHttpRequest) request).getServletRequest();
String id = servletRequest.getSession().getId();
// 標記用戶
String uid = servletRequest.getParameter("uid");
attributes.put("uid", uid);
if(uid!=null){
attributes.put("uid", uid);
System.out.println(String.format("websocket建立握手前:sessionId[%s],userId[%s]", id, uid));
}else{
return false;
}
}
return true;
}
@Override
public void afterHandshake(ServerHttpRequest serverHttpRequest, ServerHttpResponse serverHttpResponse, WebSocketHandler webSocketHandler, Exception e) {
System.out.println("握手后");
}
}
handler
public class MySocketHandler implements WebSocketHandler {
private static final Logger logger = Logger.getLogger(MySocketHandler.class);
// websocket連接建立
@Override
public void afterConnectionEstablished(WebSocketSession session) throws Exception {
logger.info("Websocket Connection established");
logger.info("getId:" + session.getId());
logger.info ("getLocalAddress:" + session.getLocalAddress().toString());
logger.info ("getUri:" + session.getUri().toString());
session.sendMessage(new TextMessage("Server:connected OK!"));
String uid = (String) session.getHandshakeAttributes().get("uid");
if (SocketSessionMap.socketSession.get(uid) == null) {
SocketSessionMap.socketSession.put(uid, session);
}
}
// 消息處理
@Override
public void handleMessage(WebSocketSession session, WebSocketMessage<?> message) throws Exception {
}
// 消息傳輸錯誤
@Override
public void handleTransportError(WebSocketSession session, Throwable throwable) throws Exception {
if (session.isOpen()) {
session.close();
}
// 移除Socket會話
Iterator<Map.Entry<String, WebSocketSession>> iter = SocketSessionMap.socketSession.entrySet().iterator();
while (iter.hasNext()) {
ConcurrentHashMap.Entry entry = iter.next();
if (entry.getValue() == session) {
SocketSessionMap.socketSession.remove(entry.getKey());
System.out.println("Socket會話已經移除:用戶ID" + entry.getKey());
break;
}
}
}
// websocket 連接關閉
@Override
public void afterConnectionClosed(WebSocketSession session, CloseStatus closeStatus) throws Exception {
// TODO Auto-generated method stub
System.out.println("Websocket:" + session.getId() + "已經關閉");
// 移除Socket會話
Iterator<Map.Entry<String, WebSocketSession>> iter = SocketSessionMap.socketSession.entrySet().iterator();
while (iter.hasNext()) {
ConcurrentHashMap.Entry entry = iter.next();
if (entry.getValue() == session) {
SocketSessionMap.socketSession.remove(entry.getKey());
System.out.println("Socket會話已經移除:用戶ID" + entry.getKey());
break;
}
}
}
@Override
public boolean supportsPartialMessages() {
return false;
}
/**
* 給某個用戶發(fā)送消息
*
* @param uid
* @param message
* @throws IOException
*/
public void sendMessageToUser(String uid, TextMessage message)
throws IOException {
WebSocketSession session = SocketSessionMap.socketSession.get(uid);
if (session != null && session.isOpen()) {
session.sendMessage(message);
}
}
}
DispatcherServlet配置
<servlet>
<servlet-name>springmvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:applicationContext-mvc.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
...
<servlet-mapping>
<servlet-name>springmvc</servlet-name>
<url-pattern>/websocket</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>springmvc</servlet-name>
<url-pattern>/sockjs/websocket/info</url-pattern>
</servlet-mapping>
<servlet>
<bean id="mySocketHandler" class="com.fingard.rh.msp.socket.MySocketHandler"/>
<!-- 握手接口/攔截器 -->
<bean id="myInterceptor" class="com.fingard.rh.msp.socket.MyInterceptor"/>
<websocket:handlers >
<websocket:mapping path="/websocket" handler="mySocketHandler"/>
<websocket:handshake-interceptors>
<ref bean="myInterceptor"/>
</websocket:handshake-interceptors>
</websocket:handlers>
<!-- sockJS -->
<websocket:handlers>
<websocket:mapping path="/sockjs/websocket/info" handler="mySocketHandler"/>
<websocket:handshake-interceptors>
<ref bean="myInterceptor"/>
</websocket:handshake-interceptors>
<websocket:sockjs />
</websocket:handlers>
問題一、問題二。請不吝賜教,萬分感激??!
北大青鳥APTECH成立于1999年。依托北京大學優(yōu)質雄厚的教育資源和背景,秉承“教育改變生活”的發(fā)展理念,致力于培養(yǎng)中國IT技能型緊缺人才,是大數(shù)據(jù)專業(yè)的國家
達內教育集團成立于2002年,是一家由留學海歸創(chuàng)辦的高端職業(yè)教育培訓機構,是中國一站式人才培養(yǎng)平臺、一站式人才輸送平臺。2014年4月3日在美國成功上市,融資1
北大課工場是北京大學校辦產業(yè)為響應國家深化產教融合/校企合作的政策,積極推進“中國制造2025”,實現(xiàn)中華民族偉大復興的升級產業(yè)鏈。利用北京大學優(yōu)質教育資源及背
博為峰,中國職業(yè)人才培訓領域的先行者
曾工作于聯(lián)想擔任系統(tǒng)開發(fā)工程師,曾在博彥科技股份有限公司擔任項目經理從事移動互聯(lián)網(wǎng)管理及研發(fā)工作,曾創(chuàng)辦藍懿科技有限責任公司從事總經理職務負責iOS教學及管理工作。
浪潮集團項目經理。精通Java與.NET 技術, 熟練的跨平臺面向對象開發(fā)經驗,技術功底深厚。 授課風格 授課風格清新自然、條理清晰、主次分明、重點難點突出、引人入勝。
精通HTML5和CSS3;Javascript及主流js庫,具有快速界面開發(fā)的能力,對瀏覽器兼容性、前端性能優(yōu)化等有深入理解。精通網(wǎng)頁制作和網(wǎng)頁游戲開發(fā)。
具有10 年的Java 企業(yè)應用開發(fā)經驗。曾經歷任德國Software AG 技術顧問,美國Dachieve 系統(tǒng)架構師,美國AngelEngineers Inc. 系統(tǒng)架構師。