文档协作技术——Operational Transformations简单了解

OT是支持协作软件系统的一种广泛使用的技术。

OT通常使用副本文档储存,每个客户端都拥有对文档的副本。客户端在本地副本以无锁非堵塞方式操作,并将改变传递到其他客户端。当客户端收到其他客户端传播的改变之后,通过转换应用更改,从而保证一致性

  1. 初始文档为"abc",并存在客户端A、B

  2. A发起操作O1=insert[0, “x”],在位置0插入字符x

    B发起操作O2=delete[2, “c”],在位置2删除字符c

在没有OT之前,加入O1发生在O2之前,那么执行O1之后字符串变为"xabc",执行O2之后由于位置2从原来的c变为了b,所以最终字符串会变为"xac",这并不符合预期

而利用OT的协作系统通常使用复制文档存储,其中每个客户端都有自己的文档副本。客户端以无锁、非阻塞的方式操作其本地副本,然后将更改传播到其他客户端。这确保了客户机在Internet等其他高延迟环境中的高响应性。

当客户端接收到从另一个客户端传播的更改时,它通常在执行更改之前对更改进行转换。转换确保所有副本都维护一致性标准(不变量)。这种操作模式产生的系统特别适合在web等高延迟环境中实现协作功能,例如同时编辑文档。

一致性模型——CCI

收敛性

所有文档副本在所有操作执行完成之后都要保持一致

意图保持

确保一个操作作用于所有文档副本的状态需与操作意图一致。操作意图定义为在操作者在其副本执行效果

因果关系保持

操作必须根据自然因果顺序执行

结构

  • transformation control algorithm:

    决定什么操作被转换,并以什么顺序转换

  • transformation properties and conditions:

    定义算法和函数之间关系,并验证算法和函数是否正确

  • transformation functions:

    负责根据操作的影响执行实际的转换。转换函数取决于操作的类型和参数,以及OT系统的数据和操作模型。

在这里插入图片描述

OT方法

以字符层面的OT方法为例

  • T({insert c1, p1}, {insert c2, p2}):

    // 例如插入b到位置4,应用插入a到位置3,那么转换之后就是插入a到位置3,因为插入b到位置4并不影响插入a到位置3
    // 如果是插入b到位置1,应用插入a到位置3,那么转换之后就是插入a到位置4,因为插入b之后,原来3的位置变成了4
    if p1 <= p2 {return insert(c1, p1)
    }else {return insert(c1, p1+1)
    }
    
  • T({insert c1, p1}, {delete, p2}):

    if p1 <= p2 {return insert(c1, p1)
    }else{return insert(c1, p1-1)
    }
    
  • T({delete, p1}, {insert c1, p2}):

    if p1 < p2 {return delete(p1)
    }else {return delete(p1+1)
    }
    
  • T({delete, p1}, {delete, p2}):

    if p1 < p2 {return delete(p1)
    }else if p1 > p2 {return delete(p1-1)
    }else{return
    }
    

设计基于字符串的转换函数比基于字符的操作更具挑战性,因为:

(1)字符串删除包括删除范围,该范围可以包括字符串中的字符以及字符之间的间隔位置;

(2)并发的字符串删除操作可能任意重叠,甚至可能与并发的插入操作重叠

(3)由前一个插入操作插入的字符串可能会被后面的插入和删除操作所改变。

上述因素使得字符转换函数的简化设计方法不适用于字符串转换函数,以获得字符串转换函数的示例)。当字符串转换函数被设计用于组撤销和一致性维护时,额外的复杂性就会发挥作用。是否支持字符串转换可能会对OT系统的各个方面(例如正确性、复杂性和效率)产生重大影响。

基本流程

介绍流程之前,先展示下client、server各自储存的信息

client

  • 最后同步版本id
  • 所有未被发送到服务器的本地改动(pending changes)
  • 所有已发送服务器的本地改动,但未ack(sent changes)
  • 当前文档状态

server

  • 所有收到但未处理的改变(pending changes)
  • 所有已处理的改变log(revision log)
  • 自上次处理后文档状态

以下便是基本流程

当客户端接收到服务端ack或者接收到操作,则版本加一

  1. 初始文档为空文本,A、B版本均为0

  2. A在本地执行{insert “hello”, @0}并发送给服务器

    A本地文本变为"hello"

  3. B在本地执行{insert “!”, @0}

    B本地文本变为"!"

  4. 随后A在本地执行{insert “world”, @5},但未发送,因为前一个条件还未ack

    A本地文本变为"helloworld"

  5. server处理完A的第一条改动,记录到revision log,随后发送ack给A,并传播改变到B

    A版本变更为1

  6. B接收到改动后,应用transformation函数作为结果。原来pending change中的{insert “!”, @0}变为{insert “!”, @5}

    B本地文本变为"hello!",并更新版本为1

  7. A发送第二次改动,B也发送改动,但A先得到ack

    此时A文本为"helloworld",B文本变为"helloworld!"

    A版本更新为2,B收到A的改动后,也更新为2

  8. server收到B的改动后,但由于B操作版本id 2小于实际id 3,因此再次应用transformation函数,并将其保存为版本3,并将最终改动传播给所有客户端

    所有客户端最终文本为"helloworld!"

基本流程中关键在于以服务器确定顺序为准,应用transformation函数转换使得最终状态一致

Undo流程

  1. 最初文档内容为"12"

  2. A执行O1 = {insert “y”, 2},随后操作传递到B,所有文档都更新为"12y"

  3. B执行O2 = {insert “x”, 0},随后操作传播到A,所有文档都更新为"x12y"

  4. A执行undo(O1)去撤回O1操作

    1. OT会先生成逆操作!O1 = {delete, 2}
    2. 然后会应用转换函数到!O1和O2,!O1’ = T(!O1, O2) = {delete, 3}

    最终得到文档"x12"

压缩

假设最初文本为"123",连续执行了

  1. O1 = {insert “x”, 2}
  2. O2 = {insert “abc”, 1}
  3. O3 = {insert “y”, 2}
  4. O4 = {delete, 7}

那么压缩过程

  1. 将最右边的操作O4与相邻的操作O3进行转置:转置(O3, O4) = [O’4, O’3],其中O’4 = Delete[6], O’3 = O3,得到L’ = [O1, O2, O’4, O3]。
  2. 将O’4进一步与新的相邻操作O2进行转置:转置(O2, O’4) = [O’ 4, O’2],其中O’4 = Delete[3], O’2 = O2,得到L’ = [O1, O’4, O2, O3]。
  3. 用新的相邻操作O1检查O’ 4;并且发现它们相互重叠互补(即对文档没有影响),因此将O’ 4和O’ 1从L中剔除,得到L’ = [O2, O3]。
  4. 将L’中相邻重叠的两个操作O2和O3合并为一个操作O’2 = Insert[1,“aYbc”],得到L’ = [O '2]。
  5. 将L’ = [O ’ 2],而不是L = [O1, O2, O3, O4]传播到各本地文档进行集成。

批判

在分布式系统世界中,操作以有限速度传播,参与者状态往往不同,因此产生的状态和操作的组合非常难以预测和理解。这使得形式证明非常复杂和容易出错

Ref

  1. https://operational-transformation.github.io/
  2. https://en.wikipedia.org/wiki/Operational_transformation
  3. https://medium.com/coinmonks/operational-transformations-as-an-algorithm-for-automatic-conflict-resolution-3bf8920ea447
  4. https://www3.ntu.edu.sg/scse/staff/czsun/projects/otfaq

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/460450.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

视频上传 - 断点续传那点事

在上一篇文章中&#xff0c;我们讲解了分片上传的实现方式。在讲解断点续传之前&#xff0c;我要把上篇文章中留下的问题讲解一下。读过上一篇文章的小伙伴们都知道&#xff0c;对于分片上传来说&#xff0c;它的传输方式分为2种&#xff0c;一种是按顺序传输&#xff0c;一种是…

#免费 苹果M系芯片Macbook电脑MacOS使用Bash脚本写入(读写)NTFS硬盘教程

Mac电脑苹果芯片读写NTFS硬盘bash脚本 &#xff08;ntfs.sh脚本内容在本文最后面&#xff09; ntfs.sh脚本可以将Mac系统(苹果M系芯片)上的NTFS硬盘改成可读写的挂载方式&#xff0c;从而可以直接往NTFS硬盘写入数据。此脚本免费&#xff0c;使用过程中无需下载任何收费软件。…

Cocos creator 3.x 刚体组件碰撞无效

Cocos creator 3.x 刚体组件碰撞无效 问题描述&#xff1a;只有一个circleCollider2D时&#xff0c;可以在碰撞时正确输出结果&#xff0c;但是当我在外围加了一个circle之后&#xff0c;期望character进入圆圈范围时就触发方法&#xff0c;此时原代码失效 import { _decorat…

【人工智能】人工智能 – 引领未来科技的潮流

写在前面 引言红利挑战结论 引言 人工智能是指使计算机系统表现出类似于人类智能的能力。其目标是实现机器具备感知、理解、学习、推理和决策等智能行为。人工智能的发展可以追溯到上世纪50年代&#xff0c;随着计算机技术和算法的不断进步&#xff0c;人工智能得以实现。 今天…

【网站项目】031网络游戏公司官方平台

&#x1f64a;作者简介&#xff1a;拥有多年开发工作经验&#xff0c;分享技术代码帮助学生学习&#xff0c;独立完成自己的项目或者毕业设计。 代码可以私聊博主获取。&#x1f339;赠送计算机毕业设计600个选题excel文件&#xff0c;帮助大学选题。赠送开题报告模板&#xff…

Java图形化界面编程—— LayoutManager布局管理器笔记

2.4 LayoutManager布局管理器 之前&#xff0c;我们介绍了Component中有一个方法 setBounds() 可以设置当前容器的位置和大小&#xff0c;但是我们需要明确一件事&#xff0c;如果我们手动的为组件设置位置和大小的话&#xff0c;就会造成程序的不通用性&#xff0c;例如&…

【golang】23、gorilla websocket 源码:examples、数据结构、流程

文章目录 一、examples1.1 echo1.1.1 server.go1.1.2 client.go 1.2 command1.2.1 功能和启动方式1.2.2 home.html1.2.3 main.go 1.3 filewatch1.3.1 html1.3.2 serveHome 渲染模板1.3.3 serveWs1.3.4 writer() 1.4 buffer pool1.4.1 server1.4.2 client 1.5 chat1.5.1 server1…

非常详细!操作系统:【文件系统概述】

&#x1f308;个人主页&#xff1a;godspeed_lucip &#x1f525; 系列专栏&#xff1a;OS从基础到进阶 1 文件系统概述1.1 文件管理应该解决的问题1.1.1 文件属性1.1.2 文件数据的组织1.1.3 文件之间的组织1.1.4 向上提供的功能1.1.5 总结 1.2 文件应该怎么存放在外存中1.3 其…

前端异步相关知识总结

目录 一、同步和异步简介 同步&#xff08;按顺序执行&#xff09; 异步&#xff08;不按顺序执行&#xff09; 异步出现的原因和需求 二、实现异步的方法 回调函数 Promise 生成器Generators/ yield async await 三、promise和 async await 区别 概念 两者的区别 …

Nacos1.X源码解读(待完善)

目录 下载源码 注册服务 客户端注册流程 注册接口API 服务端处理注册请求 设计亮点 服务端流程图 下载源码 1. 克隆git地址到本地 # 下载nacos源码 git clone https://github.com/alibaba/nacos.git 2. 切换分支到1.4.7, maven编译(3.5.1) 3. 找到启动类com.alibaba.na…

zzzzzzzzzzzzzzzzz

欢迎关注博主 Mindtechnist 或加入【Linux C/C/Python社区】一起探讨和分享Linux C/C/Python/Shell编程、机器人技术、机器学习、机器视觉、嵌入式AI相关领域的知识和技术。 磁盘满的本质分析 专栏&#xff1a;《Linux从小白到大神》 | 系统学习Linux开发、VIM/GCC/GDB/Make工具…

机器学习系列——(十六)回归模型的评估

引言 在机器学习领域&#xff0c;回归模型是一种预测连续数值输出的重要工具。无论是预测房价、股票价格还是天气温度&#xff0c;回归模型都扮演着不可或缺的角色。然而&#xff0c;构建模型只是第一步&#xff0c;评估模型的性能是确保模型准确性和泛化能力的关键环节。本文…