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首次连接时的信息,可以对其进行信息拦截,完成需求