长安链并行调度机制(1):交易调度与冲突检测流程

news2025/1/24 17:56:57

长安链采用高效的并行调度方式执行交易,了解长安链交易调度、冲突检测和DAG构建流程有助于开发者更好地理解长安链并行调度的运行机制,帮助开发者编写高质量、低冲突的智能合约,更好地构建区块链应用。

我们将用两篇文章说明长安链交易调度、冲突检测流程和DAG的构建流程。

一、 概述

1. 基础概念

交易调度执行是区块链交易执行时的核心流程,对一轮共识性能影响巨大。一笔交易执行的本质是对某些key进行读、写操作,在区块链中,一个区块中往往包含了多笔交易,交易执行完后的最新数据状态叫做“世界状态”。区块中交易调度执行共有两种方式,一种是确定性调度,比如最简单的串行执行;另一种则是非确定性调度方式,在这种方式下,为了提升调度执行性能,往往会增加交易执行的并行度。在交易并行执行下,就需要确保所有交易执行的“正确性”和“有序性”。

● 正确性:一笔交易执行过程中,所依赖的前置充分条件不能被破坏;(本质上就是读写冲突交易不能同时执行)。

● 有序性:所有节点可以按照同一个执行顺序对区块中的交易进行执行,得到一致的世界状态;(主节点需要告诉从节点按照什么顺序执行交易)。

2. 场景构造

为了方便理解,我们构造一个简单的银行借贷合约,子公司向银行申请贷款时,需要由其上级公司或总公司的存款进行担保,如果上级公司存款金额不低于贷款额度,银行会向子公司发放贷款,并立即打入到子公司账户。

图片

图1.1 世界状态

假设k1是总公司账户,k2是一级子公司账户,k3是二级子公司账户,当前的世界状态如上图所示。一个区块中共有如下4笔向银行申请贷款的交易:

图片

图 1. 区块中4笔存在冲突的交易

● tx0表示的是一级子公司k2以总公司k1存款为担保向银行申请贷款;

● tx1表示的是二级子公司k3以一级子公司k2存款为担保向银行申请贷款;

● tx2表示的是二级子公司k3以总公司k1存款为担保向银行申请贷款;

● tx3表示的是一级子公司k2以总公司k1存款为担保向银行申请贷款。

其中,tx0和tx1之间存在读写冲突,tx1和t2之间存在写写冲突,tx3和tx1之间存在写读冲突。

二、 交易冲突检测

如果区块中所有交易按序串行执行,那么所有交易执行的正确性都不会受到影响,即使两笔交易对同一个key都进行读写操作,后面的一笔交易在前一笔交易执行后的“世界状态”上执行,交易的执行结果也是正确的。

图片

图 2.1 交易串行执行

以上述的tx0和tx1为例,如果总公司k1账户余额为100元(v1),一级和二级子公司k2和k3账户余额为0元,分别以上一级公司存款额度为担保,申请贷款100元。虽然其中存在读写冲突,但是在串行执行时,后一笔交易基于前一笔交易最新的世界状态执行,两笔交易的执行结果是确定的。具体有共有上图中两种情况:

● 第一种情况,tx0先执行,贷款成功,一级子公司k2余额变为了100元,后面执行的tx1因为一级子公司k2账户余额变为100元,将会满足银行贷款条件,贷款成功。

● 第二种情况,tx1先执行,因为一级子公司k2余额开始为0,所以贷款失败,但是后面执行的tx0因为总公司k1余额还是100元,将会贷款成功。

在串行执行情况下,只会按照如上两种情况中的其中一种情况进行执行,但是无论按照哪种情况执行,交易的执行结果都是确定的,也都是符合业务逻辑的正确结果。

但是,为了提升调度执行交易的性能,采用并行调度时,情况就会复杂很多,就需要检测出来哪些并行执行的交易存在冲突(保障正确性),存在冲突的交易就需要按序执行,并且也要告知其他节点按照此顺序执行这些交易(保障有序性)。

接下来本部分将会介绍如何保障正确性,后面第二篇将介绍如何保障有序性。

1. 读写、写读冲突需要重新执行

在交易执行正确性判断时,读写冲突和写读冲突本质上是一种冲突类型,即一笔交易所依赖的读集中的内容,在执行过程中被其他交易修改了,使得这笔交易正确执行的前置充分条件被破坏了,那么该笔交易的执行结果的正确性也无法得到保障,那就需要对此笔交易重新执行。

图片

图 2.2 交易读写冲突

如上图所示,tx0和tx1在并行执行时,如果按照左图中的情况执行,不存在读写冲突,不会存在正确性的问题,因此后一笔交易不需要重新执行。但是,如果按照右图中的情况执行,tx1和tx0存在读写冲突,tx1交易执行开始时所依赖的k2,在tx1执行过程中被tx0修改了,此种情况则需要对后执行完的tx1基于tx0执行完后的最新“世界状态”重新执行,才能得到正确的结果。

图片

图 2.3 交易读写冲突(写读冲突演变的读写冲突)

如上图所示,tx1和tx3在并行执行时,如果按照图2.3左侧图中的情况执行,也不存在读写冲突,不存在正确性问题,因此后一笔交易也不用重新执行。但是,右图中的情况下,tx1和tx3之间存在读写冲突,后执行完的tx1所依赖的k2在tx1执行过程中被tx3修改了,那么后执行完的tx1就需要重新进行执行以得到正确的结果。这里着重强调一下,直观上的写读冲突,在并行执行时演变为读写冲突时,才会影响后执行完的交易的正确性。

2. 读读不冲突、写写直接覆盖

对于读读情况,我们不做过多讲解,因为在没有写的情况下,对同一个key进行并发读取,获取的值都是一致的正确结果,不存在并行执行的正确性问题。

对于写写情况,也比较简单,写写可以直接覆盖处理,直接以后一笔交易写入的值作为最终结果即可。

图片

图 2.4 写写覆盖

如上图所示,tx1和tx2都对k3进行写操作,如果按照图2.4左侧图所示tx1先执行完,tx2后执行完,k3的value变化逻辑是v3->v3'->v3'',这个执行过程是正确的。如果按照右图所示tx2先执行完,tx1后执行完,k3的value变化逻辑是v3->v3''->v3',这个执行过程也是正确的。无论采用上述哪种执行顺序,两笔交易执行过程都是符合逻辑的,正确性没有任何问题(因为两笔交易都没有对k3进行读取,不存在读写冲突),只需要保障其他节点也按照同样的顺序执行即可,这个就是另外的有序性要求了。

3、 长安链中的交易冲突检测方案

经过上面的分析,我们能够知道:并发执行下的交易冲突检测,只需要检测出来读写冲突,并对后执行完的交易重新执行,直到所有交易执行过程中均不存在读写冲突即可。

再次强调,此处的交易冲突检测,只能够在调度执行过程中确保存在读写冲突的交易能够按照某一个顺序执行得到正确的结果,仅满足了正确性条件。怎么让其他节点知道这个节点执行交易的顺序,是由有序性条件进行保障,这是由后面第二篇文章中构造DAG负责的事情,请大家不要混淆。

图片

图3.1 冲突检测流程图

在长安链中,Scheduler模块负责每个区块的调度执行,在每个区块调度执行时,Scheduler会给每个区块构造一个Snashot结构,在该结构中有四个变量需要着重关注:

● ExecutedTxs:已经执行完的交易队列,是一个Slice结构,每“正确的”执行完一笔交易,则把这笔交易append到队列中,队列索引从0开始。此外,在处理每笔交易构建simContext时,通过len(ExecutedTxs)来计算当前交易的执行编号txSeq;

● WriteTable:交易写集最新集合,是个map集合,键是交易写集中的key,值是包含写集key对应的value值和对应交易的txSeq;

● ReadTable:交易读集最新集合,也是个map集合,键是交易读集中的key,值是包含读集key对应的value值和对应交易的txSeq;

● Lock:Snapshot中的读写锁,保障并发操作下Snapshot中变量的正确性。

WriteTable和ReadTable可以理解成该区块执行的沙箱环境下,待修改和被读取的热点数据,与数据库中的数据一起构成最新的“世界状态”。

图片

图3.2 Snapshot中三种数据结构

具体流程:

1. 首先,对区块中的交易进行分发到channel通道中,此处的交易分发有一些优化,可以根据gas账户或者sender进行分组,尽可能在不同分组间实现并发调度以减少读写冲突的情况,这里不详细描述;

2. 多个goroutine协程并发地从channel中取出交易,对交易并行执行,默认设置的并发程度是4倍的cpu核数,所以该区块中的4笔交易完全是并行执行的,这里实现并发的协程池子的容量也可以根据交易冲突率进行自适应调整,这里不详细描述;

2.1 每笔交易在实际调用vm执行前,需要通过Snapshot获取当前交易的执行编号txSeq,用于构造SimContext;

2.2 此时还没有交易执行完,所以4笔获得的txSeq都是0,具体是通过len(ExecutedTxs)计算而来;

3. 4个协程并行调用vm模块对4笔交易进行执行,为每笔交易生成读写集,其实VM执行合约时也是优先从WriteTable和ReadTable进行读取,不存在再从DB中读取相关的值;

4. vm模块将包含读写集的交易执行结果返回给调度模块;

5. 调度模块对执行完的交易,进行交易冲突检测,如果同时满足如下两个条件则说明此笔交易冲突:

● 此笔交易中读集的key(假设是k0)在WriteTable中;

● 并且此笔交易的txSeq小于等于WriteTable中k0对应的txSeq;

6. 若该交易与之前已经执行完的交易不存在读写冲突,则将交易放入ExecutedTxs队列,用该交易的读写集和txSeq更新WriteTable和ReadTable;

7. 若该交易与之前已经执行完的交易存在读写冲突,则将交易重新调度分发,放入channel通道中,后面重新再次执行,直到所有交易在执行时都通过交易冲突检测。

下面分别以tx0和tx1为例讲解读写冲突具体怎么检测出来的。

图片

图3.3 冲突检测示例图

如上左图所示,此种情况不存在读写冲突,tx0和tx1并行交错执行,虽然txSeq相同,但是tx1先于tx0执行完,待tx1执行完后WriteTable中将会缓存k3对应的最新值和txSeq。等到tx0执行完时,tx0的读集k1不在WriteTable中,即tx0交易执行所依赖的前置充分条件没有被破坏,tx0执行正确,此时将tx0 添加进ExecutedTxs队列,用该交易的读写集和txSeq更新WriteTable和ReadTable即可。

如上右图,此种情况存在读写冲突,tx0先于tx1执行完,待tx0执行完后WriteTable中将会缓存k2对应的最新值和txSeq。等到tx1执行完时,tx1的读集k2在WriteTable中,并且tx1的txSeq为0小于等于WriteTable中k2当前的最新执行序0,tx1交易执行所依赖的前置充分条件被破坏,tx1执行失败,此时则需要对tx1重新调度分发,再重新执行。

图片

图3.4 冲突后重新执行

当tx1重新执行时,假设只有这一笔交易执行,那么其txSeq的值将为len(ExecutedTxs)的值为1,随后VM在执行交易时,会优先从Snapshot的WriteTable和ReadTable中读取k2的值,此时读取的结果是最新的v2',待tx1执行完后,检测k2虽然在WriteTable中,但是其txSeq为1大于WriteTable中的txSeq 0,所以不冲突,将tx1添加进ExecutedTxs队列,并用该交易的读写集和txSeq更新WriteTable和ReadTable即可。

4. 结语

至此我们介绍了长安链中交易调度和交易冲突检测流程及机制,此时确保了交易执行过程的正确性,我们将在下一篇文章中介绍,长安链如何实现交易执行的有序性。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Linux特殊指令

目录 1.dd命令 2.mkfs格式化 3.df命令 4.mount实现硬盘的挂载 5.unshare 1.dd命令 dd命令可以用来读取转换并输出数据。 示例一: if表示infile,of表示outfile。这里的/dev/zero是一个特殊文件,会不断产生空白数据。 bs表示复制一块的大…

ZLMediaKit 重建docker包

1.下载容器到本地服务器并运行 #此镜像为github持续集成自动编译推送,跟代码(master分支)保持最新状态 docker run -id -p 1935:1935 -p 8080:80 -p 8443:443 -p 8554:554 -p 10000:10000 -p 10000:10000/udp -p 8000:8000/udp -p 9000:9000/udp zlmediakit/zlmedi…

阿姆达尔定律(Amdahl‘s Law)通俗解释

阿姆达尔定律(Amdahl’s Law),它描述了在对系统的某个部分进行加速时,该部分对整体系统性能的影响,取决于该部分的重要性和加速程度。 原书给的例子不太好懂,下面是一个更好懂的例子。 例子:汽车…

Python数据分析实战-修改 DataFrame 中的字段(列)名(附源码和实现效果)

实现功能 修改 DataFrame 中的字段(列)名 实现代码 import pandas as pd# 创建一个示例 DataFrame df pd.DataFrame({A: [1, 2, 3], B: [4, 5, 6], C: [7, 8, 9]})# 使用 rename 方法修改列名这,将返回一个新的 DataFrame,其中列名已更改…

构建产品帮助中心:SaaS产品帮助中心必备要素,需要规避的问题

帮助中心本来是为了帮助客户解决问题,了解产品价值,例如:产品使用手册、新手入门视频、核心功能的操作演示、常见问题FAQ都可以当作产品帮助中心的范畴。 以SaaS产品为例,开发帮助中心需要具备内容展示、搜索查询、文档理解等三大…

三分钟了解车规级芯片的特点

汽车已经不再只是一种交通工具,而是一个复杂的智能系统。现代汽车配备了许多电子设备和系统,如发动机控制单元(ECU)、安全气囊系统、防抱死刹车系统(ABS)、自动驾驶功能、娱乐系统等。这些系统需要可靠的电…

蒲公英路由器如何设置远程打印?

现如今,打印机已经是企业日常办公中必不可少的设备,无论何时何地,总有需要用到打印的地方,包括资料文件、统计报表等等。 但若人在外地或分公司,有文件急需通过总部的打印机进行打印时,由于不在同一物理网络…

MFC网络编程简单例程

目录 一、关于网络的部分概念1 URL(网址)及URL的解析2 URL的解析3 域名及域名解析3 IP及子网掩码4 什么是Web服务器5 HTTP的基本概念6 Socket库概念7 协议栈8 Socket库收发数据基本步骤 二、基于TCP的网络应用程序三、基于UDP的网络应用程序 一、关于网络的部分概念 1 URL(网址…

低代码的探索之路

Gartner发布报告指出,2023年全球低代码开发平台市场规模将达到345亿美元,比2022年增长20%。 目前,国内外已经有许多低代码平台,包括OutSystems、Mendix、Appian、Microsoft Power App等。这些平台提供了丰富的功能和工具&#xff…

故障演练实战

做了两年多的故障演练, 一直想聊聊自己对故障演练的理解,但是每次提起笔都不知道写一些什么。 什么是故障演练 为什么要做故障演练 在没做故障演练之前,我想很多人可能和我有一样的想法,我的系统跑了好多年,也没出什么…

Docker容器数据持久化

Docker容器数据持久化 Docker容器数据卷:volumes 1、什么是数据卷? 数据卷是经过特殊设计的目录,可以绕过联合文件系统,为一个或者多个容器提供访问,数据卷设计的目的,在于数据的永久存储,它完…

Java-day11(集合)

集合 1.集合框架 用于存储数量不等的多个对象,还可用于保存具有映射关系的关联数组 Java集合可分为Collection和Map两种体系 Collection接口 Set:元素无序,不可重复的集合----(类似数学中的集合) List:元素有序,可重…

linux内网yum源服务器搭建

1.nginx: location / {root /usr/local/Kylin-Server-V10-SP3-General-Release-2303-X86_64;autoindex on;autoindex_localtime on;autoindex_exact_size off; } 注:指定到镜像的包名 2.修改yum源地址 cd /etc/yum.repos.d/vim kylin_x86_64.repo 注: --enabled设置为1 3.重…

Android Studio 的github 工程克隆

上文介绍了Android Studio 里的"Git 建立和简单操作。本文介绍从github 上的工程fork 和clone到本地,然后学习和改进。 本文参考 https://learntodroid.com/how-to-use-git-and-github-in-android-studio/ 克隆clone Github 仓库: 先 Fork 你选择…

仓储23代拣货标签操作指导

服务器使用 V1.4基站已经内置服务程序,无需搭建服务;可跳至第1.4部分 服务器搭建 安装mysql5.7, 创建db_wms数据库并导入原始数据库文件 安装jdk1.8, 配置java环境变量 下载tomca8.0, 部署wms.war到tomcat, 并启动tomcat 下载资源 Windows 64bit: …

知乎如何精准引流?

知乎,用过的人都知道,它是一个相当重要的引流平台。因为它用户规模大、粘性高、活跃性强、百度权重高,流量也相对精准,这也意味者变现能力强。 做引流的朋友都知道,想要把用户从别的平台引流到自己微信上,就…

Gradle项目如何开启debug?

一、打开Idea的终端,输入命令gradle build --debug 开启构建DeBug模式 二、点击编辑配置,并填写自己debug的名称 三、编辑好debug配置后,点击以下debug按钮,就可以开始调试代码了。

vue3下的密码输入框(antdesignvue)

参考:vue下的密码输入框 注意:这是个半成品,有些问题(输入到第6位的时候会往后窜出来一个空白框、光标位置会在数字前面),建议不采用下面这种方式,用另外的(画六个input框更方便) 效果预览 实现思路 制作6个小的正方形div 用一个input覆盖在6个div上 给input设置透明(…

LabVIEW开发油气井管道内无线通信微波系统

LabVIEW开发油气井管道内无线通信微波系统 由于石油和天然气行业的重要性,许多公司和研究人员正在研究和开发新的通信和传感器技术。需要解决的问题与管道状况、地震活动、腐蚀水平、可能的气体泄漏检测和其他性能问题的实时和连续监测有关。处理这些问题的最广泛使…

forlium 笔记 Map

用于创建交互式地图 1 主要参数 1.1. location 地图位置 地图的经纬度 import foliumm folium.Map(location[31.186358, 121.510256],zoom_start15)m 1.2 tiles 内置样式 默认是OpenStreetMap 1.2.1 Stamen Terrain 它强调了地形特征,如山脉、河流和道路 m …