微信小程序结合SpringBoot实现WebSocket长链接
最近在做有关前后端的项目,前端主要是用Vue框架和微信小程序的原生框架
后端主要是采用Flask或SpringBoot
引入
我们知道页面的数据会在页面加载的时候触发created,mounted,onload 等方法去后端获取数据,那么现在有一个需求,我们要做一个聊天软件,页面不刷新我们就拿不到我们想要的数据,怎么才能实时通讯呢?
还有一种需求,一个订单系统,客户发起订单后,商家难倒要一直刷新页面才能实时查看有没有新的订单吗
添加好友的模块
一些运动软件实时反馈步数等等,如果是基于前后端的web开发,那么我们可以使用WebSocket 来实现前面所说的需求
WebSocket
它和Http一样是一个网络请求的协议,与Http不一样的是WebSocket是一个持久化通信协议,
传统的Http协议:发送请求获取响应(request-response)后来加入了keep-alive即在一次http链接中可以发送多次请求获取多次响应,
很明显http的协议的响应都是被动的。
WebSocket可以实现主动向前端传递消息
具体内容可以查看这篇博客:https://blog.csdn.net/frank_good/article/details/50856585
Http:
客户端:有消息吗
服务器:没有
客户端:有消息吗
服务器:没有
客户端:有消息吗
服务器:有,消息是xxxxx
websocket:
客户端:有消息和我说
服务器:新消息:xxxx
服务器:新消息:xxxx
服务器:新消息:xxxx
微信小程序部分实现
小程序内置了websocket的api,我们直接调用即可
websocket可以放在全局,也可以放在单独的某个页面,具体看需求,我实现的是单独一个页面的
js部分
Page({
data: {
inputStr: '',
getStr:'',
socketStatus: 'closed',
SocketTask:null,
lockReconnect: false,
wsCreateHandler: null,
timeoutObj:null,
serverTimeoutObj:null,
},
extraLine: [],
add() {
this.extraLine.push(this.getStr)
this.setData({
text: this.extraLine.join('\n'),
})
setTimeout(() => {
this.setData({
scrollTop: 99999
})
}, 0)
},
onLoad: function (options) {
let that = this
if (that.data.socketStatus === 'closed') {
that.openSocket();
}
},
openSocket() {
try {
this.SocketTask = wx.connectSocket({
url: "ws://xxxxxxxxxxxx",
})
} catch (e) {
this.ReConnect()
}
this.SocketTask.onOpen(() => {
console.log('WebSocket 已连接')
this.startHeartCheck()
this.socketStatus = 'connected';
})
this.SocketTask.onClose(() => {
console.log('WebSocket 已断开')
this.socketStatus = 'closed'
this.ReConnect()
})
this.SocketTask.onError(error => {
this.socketStatus = 'closed'
this.ReConnect()
})
this.SocketTask.onMessage(message => {
console.log(message)
this.startHeartCheck()
this.getStr = message.data
let jsonObj = JSON.parse(this.getStr)
this.getStr = jsonObj.message
let res = jsonObj.type
if (res === "pong")
return
this.add()
})
},
closeSocket() {
if (this.socketStatus === 'connected') {
this.SocketTask.close({
success: () => {
this.socketStatus = 'closed'
}
})
}
},
sendMessage() {
if (this.socketStatus === 'connected') {
let jsonObj = {formUserId:"A",toUserId:"B",message:this.inputStr}
let jsonStr = JSON.stringify(jsonObj)
this.SocketTask.send({
data: jsonStr,
})
}
},
InputStr: function(e) {
this.inputStr = e.detail.value
},
btnFun:function(){
this.sendMessage()
},
ReConnect(){
if (this.lockReconnect)
return
console.log("重新连接...")
this.lockReconnect = true
this.wsCreateHandler && clearTimeout(this.wsCreateHandler)
this.wsCreateHandler = setTimeout(()=>{
this.openSocket()
this.lockReconnect = false
},2000)
},
resetHeartCheck(){
clearTimeout(this.timeoutObj)
clearTimeout(this.serverTimeoutObj)
this.startHeartCheck()
},
startHeartCheck(){
this.timeoutObj && clearTimeout(this.timeoutObj)
this.serverTimeoutObj && clearTimeout(this.serverTimeoutObj)
this.timeoutObj = setTimeout(()=>{
console.log("发送ping到服务器")
try {
this.SocketTask.send({
data: "ping"
})
} catch(e) {
console.log("发送ping失败")
}
this.serverTimeoutObj = setTimeout(()=>{
this.ReConnect()
},15000)
},15000)
}
})
页面部分
<text>输入文字</text>
<input class="weui-input" auto-focus placeholder="文本输入框" bindinput="InputStr"/>
<button bindtap="btnFun">确定输入</button>
<view scroll-y="true" scroll-top="{{scrollTop}}">
<text>{{text}}</text>
</view>
后端SpringBoot实现
WebSocketConfig.java
package xihema.websocket.car.config;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.socket.server.standard.ServerEndpointExporter;
@Configuration
public class WebSocketConfig {
@Bean
public ServerEndpointExporter serverEndpointExporter() {
return new ServerEndpointExporter();
}
}
WebSocketEndPoint.java
package xihema.websocket.car.websocket;
import com.alibaba.fastjson.JSON;
import org.springframework.stereotype.Component;
import javax.websocket.OnClose;
import javax.websocket.OnMessage;
import javax.websocket.OnOpen;
import javax.websocket.Session;
import javax.websocket.server.PathParam;
import javax.websocket.server.ServerEndpoint;
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;
@ServerEndpoint(value = "/websocket/{userId}")
@Component
public class WebSocketEndPoint {
private Session session;
@OnOpen
public void onOpen(Session session, @PathParam("userId") String userId) {
SessionPool.sessions.put(userId, session);
}
@OnClose
public void onClose(Session session) throws IOException {
SessionPool.close(session.getId());
session.close();
}
@OnMessage
public void onMessage(String message, Session session) {
System.out.println(message);
if (message.equalsIgnoreCase("ping")) {
try {
Map<String, Object> params = new HashMap<>();
params.put("type", "pong");
session.getBasicRemote().sendText(JSON.toJSONString(params));
} catch (Exception e) {
e.printStackTrace();
}
return;
}
SessionPool.sendMessage(message);
}
}
SessionPool.java
package xihema.websocket.car.websocket;
import javax.websocket.Session;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
public class SessionPool {
public static Map<String, Session> sessions = new ConcurrentHashMap<>();
public static void close(String sessionId) {
for (String userId : SessionPool.sessions.keySet()) {
Session session = SessionPool.sessions.get(userId);
if (session.getId().equals(sessionId)) {
sessions.remove(userId);
break;
}
}
}
public static void sendMessage(String userId, String message) {
sessions.get(userId).getAsyncRemote().sendText(message);
}
public static void sendMessage(String message) {
for (String sessionId : SessionPool.sessions.keySet()) {
SessionPool.sessions.get(sessionId).getAsyncRemote().sendText(message);
}
}
public static void sendMessage(Map<String, Object> params) {
String userId = params.get("formUserId").toString();
String toUserId = params.get("toUserId").toString();
String msg = params.get("message").toString();
Session session = sessions.get(toUserId);
if (session != null) {
session.getAsyncRemote().sendText(msg);
}
}
}
代码部分功能分析
重连机制
当当前页面的websocket与后端链接断开时,或报错或链接的时候遇到问题,没有连接成功,如果没有重连机制,那么客户端和服务器的交互到此结束,但是事实上并非如此,如果服务器又恢复正常了,能连接上了呢,前端只能刷新页面才能连上,重连机制就是在服务器断开后不停进行重连的操作。
呈现的效果:
心跳机制
当客户端和浏览器之间的网络连接中断,客户端和服务器都无法发现报错,但是此时客户端和服务器已经不能通信,添加心跳机制能很好的避免这个问题,和重连机制类似,每隔几秒钟向服务器发送一次消息,服务器能给出答复说明他们之间连接没问题,否则进行重连
心跳重连一般添加到链接后和发消息后:
显示查看心跳机制:
写在最后
第一次接触websocket如果代码中存在问题请指出!!
|