1、协同编辑的意思是什么?
其实,协同编辑无非就是字面意思,多人同时编辑,并且能够同步看到对方问保存的数据,典型的例子可以参考石墨文档,腾讯文档。
2、技术解决
核心技术就是信息的实时通信
以及多人编辑时所产生的冲突
这里我采用websocket来进行实时通信,大家都知道他是一个全双工通信协议,经过时间的考证,还是非常好用的,多数流行语言都有与之响应封装好的软件包
它的最大特点就是,服务器可以主动向客户端推送信息,客户端也可以主动向服务器发送信息,是真正的双向平等对话,属于服务器推送技术的一种。详细的概念可以百度自行查找,比比皆是
编辑冲突的问题可以使用合并算法,上锁等技术(这里没有做过多的研究,所以我使用下面的方式,嘻嘻)
编辑冲突问题交给用户,当前用户实时地看到了别人正在编辑,那么当前用户就自觉性地停止编辑。
3、实现思路
1. 用户打开图像编辑页面,与后端建立长连接。
2. 后端将当前用户加入当前图像编辑列表。
3. 前端监听用户对于图像内容的修改,每一次修改将整个修改内容发送给后端。
4. 后端接收到信息,不做任何处理,直接将图像信息发送给图像编辑用户列表中其他的所有用户。
5. 前端收到后端的文本信息直接覆盖掉当前图像内容。
4、示例
废话就不多说了,直接上代码(复制粘贴即可使用呦)
这里是使用java语言编写的,采用的是原生注解方式,还有其他实现方式,在这里不一一介绍了(主要是上百度搜的没几个能用的,不是相关代码不全,就是长篇大论,最后还是不行,我实在是不会用啊!!!)
首先我们要使用websocket,肯定是要在pom里导入依赖包的(maven无法解析的,可以添加一下响应的版本号)
<!-- webscoket -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-websocket</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
创建MyWebSocketConfig文件来注入bean
package com.example.javawebsocket;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.socket.server.standard.ServerEndpointExporter;
//@Configuration注解标识的类中声明了1个或者多个@Bean方法,Spring容器可以使用这些方法来注入Bean
@Configuration
public class MyWebSocketConfig {
@Bean
public ServerEndpointExporter serverEndpointExporter() {
return new ServerEndpointExporter();
}
}
接下来就是创建一个原生注解的文件
package cn.staitech.system.utils;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.ObjectOutputStream;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.CopyOnWriteArraySet;
import javax.websocket.*;
import javax.websocket.server.PathParam;
import javax.websocket.server.ServerEndpoint;
import cn.staitech.common.security.utils.SecurityUtils;
import com.alibaba.fastjson.parser.JSONToken;
import io.swagger.models.auth.In;
import org.springframework.beans.BeanUtils;
import org.springframework.stereotype.Component;
@Component
//主要是将目前的类定义成一个websocket服务器端, 注解的值将被用于监听用户连接的终端访问URL地址,客户端可以通过这个URL来连接到WebSocket服务器端
@ServerEndpoint(value = "/websocket/{imageid}")
//此注解相当于设置访问URL
public class WebSocketServer {
/**
* 与某个客户端的连接会话,需要通过它来给客户端发送数据
*/
private Session session;
private String userName;
/** concurrent包的线程安全Set,用来存放每个客户端对应的CumWebSocket对象。*/
private static CopyOnWriteArraySet<WebSocketServer> webSockets =new CopyOnWriteArraySet<>();
/**为了保存在线用户信息,在方法中新建一个list存储一下【实际项目依据复杂度,可以存储到数据库或者缓存】**/
private static Map<Long,Session> sessionPool = new HashMap<Long,Session>();
private static Map<Integer, CopyOnWriteArraySet<WebSocketServer>> newwebSockets = new HashMap<Integer,CopyOnWriteArraySet<WebSocketServer>>();
/**
* 建立连接
* @param session
* @param userName
*/
@OnOpen
public void onOpen(Session session,@PathParam(value = "imageid") Integer imageid) throws IOException {
Long userid = SecurityUtils.getUserId();
this.session = session;
webSockets.add(this);
// 如果id不存在,创建一个新的用户存储池,格式为 图像id:[用户1,用户2]
if(! newwebSockets.containsKey(imageid)){
CopyOnWriteArraySet<WebSocketServer> webSocketslist =new CopyOnWriteArraySet<>();
newwebSockets.put(imageid,webSocketslist);
newwebSockets.get(imageid).add(this);
}else{
newwebSockets.get(imageid).add(this);
}
sessionPool.put(userid, session);
Session res = sessionPool.get(userid);
System.out.println(imageid+"【websocket消息】有新的连接,总数为:"+newwebSockets.get(imageid).size());
}
/**
* 断开连接
*/
@OnClose
public void onClose(@PathParam(value = "imageid") Integer imageid) {
webSockets.remove(this);
newwebSockets.get(imageid).remove(this);
System.out.println(imageid + "【websocket消息】连接断开,总数为:"+newwebSockets.get(imageid).size());
}
/**
* 发送错误
* @param session
* @param error
*/
@OnError
public void onError(Session session, Throwable error) {
System.out.println("[连接ID:{}] 错误原因:{}" + this.session + error.getMessage());
}
/**
* 收到信息
*/
@OnMessage
public String onMessage(String message) {
System.out.println("【websocket消息】收到客户端消息:"+message);
return message;
}
// 此为广播消息
public void sendAllMessage(String message,Integer imageid) {
for(WebSocketServer webSocket : newwebSockets.get(imageid)) {
System.out.println("【websocket消息】广播消息:"+message);
try {
webSocket.session.getAsyncRemote().sendText(message);
} catch (Exception e) {
e.printStackTrace();
}
}
}
// 此为单点消息
public void sendOneMessage(String userName, String message) {
System.out.println("【websocket消息】单点消息:"+message);
System.out.println(sessionPool);
Session session = sessionPool.get(userName);
if (session != null) {
try {
session.getAsyncRemote().sendText(message);
} catch (Exception e) {
e.printStackTrace();
}
}
}
}
我们我两个接口来调用一下上面两个接口
package com.example.javawebsocket;
import com.example.javawebsocket.WebSocketServer;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;
@RestController
@RequestMapping("/annotation")
public class Websocket {
@Autowired
private WebSocketServer webSocketServer;
// @GetMapping("/sendAllWebSocket")
// public String test() {
// String text="你们好!这是websocket群体发送!";
// webSocketServer.sendAllMessage(text);
// return text;
// }
@GetMapping("/sendOneWebSocket/{userName}")
public String sendOneWebSocket(@PathVariable("userName") String userName) {
String text=userName+" 你好! 这是websocket单人发送!";
webSocketServer.sendOneMessage(userName,text);
return text;
}
@CrossOrigin
@RequestMapping(value = "/newimage",method = RequestMethod.POST)
public String newimage(@RequestBody socketvo req){
webSocketServer.sendAllMessage(req.getImagename(), req.getImageid());
System.out.println(req.getImagename());
return "ok";
}
}
基本上大致就是酱紫了,可以根据不同需求来自行更改
下面使用netty的形式来实现websocket
创建global包,在包下创建ChannelSupervise用于定义管道集合
import io.netty.channel.Channel;
import io.netty.channel.ChannelId;
import io.netty.channel.group.ChannelGroup;
import io.netty.channel.group.DefaultChannelGroup;
import io.netty.handler.codec.http.websocketx.TextWebSocketFrame;
import io.netty.util.concurrent.GlobalEventExecutor;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
public class ChannelSupervise {
private static ChannelGroup GlobalGroup = new DefaultChannelGroup(GlobalEventExecutor.INSTANCE);
private static ConcurrentMap<String, ChannelId> ChannelMap = new ConcurrentHashMap();
public static final ConcurrentMap<Channel, Long> CHANNEL_MAP = new ConcurrentHashMap<>();
public static void addChannel(Channel channel) {
GlobalGroup.add(channel);
ChannelMap.put(channel.id().asShortText(), channel.id());
}
public static void removeChannel(Channel channel) {
GlobalGroup.remove(channel);
ChannelMap.remove(channel.id().asShortText());
}
public static Channel findChannel(String id) {
return GlobalGroup.find(ChannelMap.get(id));
}
public static void send2All(TextWebSocketFrame tws) {
GlobalGroup.writeAndFlush(tws);
}
public static void addChannelTest(Channel channel, Long slideId) {
CHANNEL_MAP.put(channel, slideId);
System.out.println(CHANNEL_MAP);
}
public static void removeChannelTest(Channel channel) {
CHANNEL_MAP.remove(channel);
}
}
创建文件Chatgroup
import io.netty.channel.Channel;
import io.netty.channel.group.ChannelGroup;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
public class ChatGroup {
public static ConcurrentMap<String, ChannelGroup> chatGroupMap = new ConcurrentHashMap<>();
public static final ConcurrentMap<Integer, Channel> CHANNEL_MAP = new ConcurrentHashMap<>();
}
创建websocket包,在其包下创建
NioWebSocketServerNio文件
import io.netty.bootstrap.ServerBootstrap;
import io.netty.channel.Channel;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.nio.NioServerSocketChannel;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Component;
@Slf4j
@Component
public class NioWebSocketServer {
@SuppressWarnings("checkstyle:MissingJavadocMethod")
public void start() {
log.info("正在启动websocket服务器");
NioEventLoopGroup boss = new NioEventLoopGroup();
NioEventLoopGroup work = new NioEventLoopGroup();
try {
ServerBootstrap bootstrap = new ServerBootstrap();
bootstrap.group(boss, work);
bootstrap.channel(NioServerSocketChannel.class);
bootstrap.childHandler(new NioWebSocketChannelInitializer());
Channel channel = bootstrap.bind(9999).sync().channel();
log.info("webSocket服务器启动成功:" + channel);
channel.closeFuture().sync();
} catch (InterruptedException e) {
e.printStackTrace();
log.info("运行出错:" + e);
Thread.currentThread().interrupt();
} finally {
boss.shutdownGracefully();
work.shutdownGracefully();
log.info("websocket服务器已关闭");
}
}
}
其中9999是端口号,可自行更改
创建NioWebSocketChannelInitializer文件用于初始化
import io.netty.channel.ChannelInitializer;
import io.netty.channel.socket.SocketChannel;
import io.netty.handler.codec.http.HttpObjectAggregator;
import io.netty.handler.codec.http.HttpServerCodec;
import io.netty.handler.logging.LoggingHandler;
import io.netty.handler.stream.ChunkedWriteHandler;
/**
* .
*/
public class NioWebSocketChannelInitializer extends ChannelInitializer<SocketChannel> {
@Override
protected void initChannel(SocketChannel ch) {
//设置log监听器,并且日志级别为debug,方便观察运行流程
ch.pipeline().addLast("logging", new LoggingHandler("DEBUG"));
//设置解码器
ch.pipeline().addLast("http-codec", new HttpServerCodec());
//聚合器,使用websocket会用到
ch.pipeline().addLast("aggregator", new HttpObjectAggregator(65536));
//用于大数据的分区传输
ch.pipeline().addLast("http-chunked", new ChunkedWriteHandler());
//自定义的业务handler
ch.pipeline().addLast("handler", new NioWebSocketHandler());
}
}
使用NioWebSocketHandler文件来进行对于消息之间的处理
import cn.staitech.anno.netty.global.ChannelSupervise;
import cn.staitech.anno.netty.global.ChatGroup;
import cn.staitech.anno.domain.vo.AnnotationBroadcastVO;
import cn.staitech.anno.domain.vo.AnnotationPageVO;
import cn.staitech.anno.domain.vo.BroadcastVO;
import cn.staitech.anno.domain.vo.SendAllMessageVO;
import cn.staitech.anno.service.AnnotationService;
import com.alibaba.fastjson.JSON;
import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled;
import io.netty.channel.Channel;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelFutureListener;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.SimpleChannelInboundHandler;
import io.netty.handler.codec.http.DefaultFullHttpResponse;
import io.netty.handler.codec.http.FullHttpRequest;
import io.netty.handler.codec.http.HttpResponseStatus;
import io.netty.handler.codec.http.HttpVersion;
import io.netty.handler.codec.http.websocketx.CloseWebSocketFrame;
import io.netty.handler.codec.http.websocketx.PingWebSocketFrame;
import io.netty.handler.codec.http.websocketx.PongWebSocketFrame;
import io.netty.handler.codec.http.websocketx.TextWebSocketFrame;
import io.netty.handler.codec.http.websocketx.WebSocketFrame;
import io.netty.handler.codec.http.websocketx.WebSocketServerHandshaker;
import io.netty.handler.codec.http.websocketx.WebSocketServerHandshakerFactory;
import io.netty.util.CharsetUtil;
import lombok.extern.slf4j.Slf4j;
import org.springframework.amqp.core.Message;
import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.amqp.support.AmqpHeaders;
import org.springframework.messaging.handler.annotation.Header;
import org.springframework.messaging.handler.annotation.Payload;
import org.springframework.stereotype.Component;
import javax.annotation.Resource;
import java.nio.CharBuffer;
import java.util.Date;
import java.util.List;
import java.util.Map;
import static io.netty.handler.codec.http.HttpUtil.isKeepAlive;
@Slf4j
@Component
public class NioWebSocketHandler extends SimpleChannelInboundHandler<Object> {
private static AnnotationService annotationService;
/**
* . 注入的时候,给类的 service 注入 ,直接使用装饰符进行装饰会报错 空指针异常
*/
@Resource
public void setUserService(AnnotationService annotationService) {
NioWebSocketHandler.annotationService = annotationService;
}
private WebSocketServerHandshaker handshaker;
/**
* . 使用channelRead0不用释放资源,jvm会自动释放
*/
@Override
protected void channelRead0(ChannelHandlerContext ctx, Object msg) throws Exception {
log.debug("收到消息:" + msg);
// 测试msg是否是FullHttpRequest的类的实例 FullHttpRequest:请求参数中的信息
if (msg instanceof FullHttpRequest) {
//以http请求形式接入,但是走的是websocket
handleHttpRequest(ctx, (FullHttpRequest) msg);
} else if (msg instanceof WebSocketFrame) {
//处理websocket客户端的消息
handlerWebSocketFrame(ctx, (WebSocketFrame) msg);
}
}
@Override
public void channelActive(ChannelHandlerContext ctx) throws Exception {
//添加连接
log.debug("客户端加入连接:" + ctx.channel());
ChannelSupervise.addChannel(ctx.channel());
}
@Override
public void channelInactive(ChannelHandlerContext ctx) throws Exception {
//断开连接
log.debug("客户端断开连接:" + ctx.channel());
ChannelSupervise.removeChannel(ctx.channel());
ChannelSupervise.removeChannelTest(ctx.channel());
}
@SuppressWarnings("checkstyle:MissingJavadocMethod")
public void delete(Channel channel) {
// 根据channel.id 删除map中的元素
ChatGroup.CHANNEL_MAP.values().removeIf(value -> value.toString().contains(channel.toString()));
}
// 数据发送时根据键来判断发送给谁
/**
* . 循环map里面的每一对键值对,然后获取key和value 判断value是否等于slide,若是相等,就进行发送
*/
public static void sendAll(Long slideId, BroadcastVO message) {
JSON json = (JSON) JSON.toJSON(message);
for (Map.Entry<Channel, Long> vo : ChannelSupervise.CHANNEL_MAP.entrySet()) {
if (vo.getValue().equals(slideId)) {
TextWebSocketFrame tws = new TextWebSocketFrame(String.valueOf(json));
vo.getKey().writeAndFlush(tws);
}
}
}
@Override
public void channelReadComplete(ChannelHandlerContext ctx) throws Exception {
ctx.flush();
}
private void handlerWebSocketFrame(ChannelHandlerContext ctx, WebSocketFrame frame) {
// 判断是否关闭链路的指令
if (frame instanceof CloseWebSocketFrame) {
handshaker.close(ctx.channel(), (CloseWebSocketFrame) frame.retain());
return;
}
// 判断是否ping消息
if (frame instanceof PingWebSocketFrame) {
ctx.channel().write(new PongWebSocketFrame(frame.content().retain()));
return;
}
// 本例程仅支持文本消息,不支持二进制消息
if (!(frame instanceof TextWebSocketFrame)) {
log.debug("本例程仅支持文本消息,不支持二进制消息");
throw new UnsupportedOperationException(
String.format("%s frame types not supported", frame.getClass().getName()));
}
// 返回应答消息
String request = ((TextWebSocketFrame) frame).text();
log.debug("服务端收到:" + request);
TextWebSocketFrame tws = new TextWebSocketFrame(new Date().toString() + ctx.channel().id() + ":" + request);
// 群发
ChannelSupervise.send2All(tws);
// 返回【谁发的发给谁】
// ctx.channel().writeAndFlush(tws);
}
/**
* . 唯一的一次http请求,用于创建websocket
*/
private void handleHttpRequest(ChannelHandlerContext ctx, FullHttpRequest req) {
//要求Upgrade为websocket,过滤掉get/Post
if (!req.decoderResult().isSuccess() || (!"websocket".equals(req.headers().get("Upgrade")))) {
//若不是websocket方式,则创建BAD_REQUEST的req,返回给客户端
sendHttpResponse(ctx, req,
new DefaultFullHttpResponse(HttpVersion.HTTP_1_1, HttpResponseStatus.BAD_REQUEST));
return;
}
WebSocketServerHandshakerFactory wsFactory = new WebSocketServerHandshakerFactory(
"ws:/" + ctx.channel() + "/websocket", null, false);
long slideId = Long.parseLong(req.getUri().split("/")[req.getUri().split("/").length - 1]);
ChannelSupervise.addChannelTest(ctx.channel(), slideId);
handshaker = wsFactory.newHandshaker(req);
if (handshaker == null) {
WebSocketServerHandshakerFactory.sendUnsupportedVersionResponse(ctx.channel());
} else {
handshaker.handshake(ctx.channel(), req);
// 进行函数发送
connectSend(slideId, ctx);
}
}
/**
* . 拒绝不合法的请求,并返回错误信息
*/
private static void sendHttpResponse(ChannelHandlerContext ctx, FullHttpRequest req, DefaultFullHttpResponse res) {
// 返回应答给客户端
if (res.status().code() != 200) {
ByteBuf buf = Unpooled.copiedBuffer(res.status().toString(), CharsetUtil.UTF_8);
res.content().writeBytes(buf);
buf.release();
}
ChannelFuture f = ctx.channel().writeAndFlush(res);
// 如果是非Keep-Alive,关闭连接
if (!isKeepAlive(req) || res.status().code() != 200) {
f.addListener(ChannelFutureListener.CLOSE);
}
}
static void connectSend(Long slideId, ChannelHandlerContext ctx) {
// 连接成功进行发送
TextWebSocketFrame tws = new TextWebSocketFrame(String.valueOf(slideId));
ctx.channel().writeAndFlush(tws);
}
}
}
因需求较为特殊,需要在首次连接时发送数据库中数据,如有需要,可以看hansleHttpRequest函数,函数中包含了web首次连接时的信息,可以对其进行信息拦截,完成需求