当前位置: 首页 > article >正文

Flutter异步原理-Future

前言

  • 在 Dart 中,谈到异步就离不开 Future。无论是 .then()、还是 await,它们背后运作的都是一个私有实现类:_Future ,我们平时使用的 Future 只是一个抽象接口,其真正的实现逻辑由_Future 承担。
class _Future<T> implements Future<T> {int _state; // 当前状态,如未完成、已完成、链式等待等。_Zone _zone; // 创建时绑定的执行 Zone。var _resultOrListeners; // 保存完成结果或注册的监听器链表。bool get _mayComplete => (_state & _completionStateMask) == _stateIncomplete;bool get _isComplete => (_state & (_stateValue | _stateError)) != 0;bool get _hasError => (_state & _stateError) != 0;void _complete(FutureOr<T> value); // 完成 Future(同步)void _asyncComplete(FutureOr<T> value); // 异步完成 Future(microtask)void _chainCoreFutureSync(_Future source, _Future target); // 同步触发 Future,如果source已完成,则把结果克隆给target,并触发回调。如果source未完成,void _chainCoreFutureAsync(Future source); // 异步链接 Futurevoid _chainForeignFuture(Future source); // 链接外部 Futurevoid _propagateToListeners();//将Future的完成结果(值或错误)传播给所有已注册的监听器void _addListener(_FutureListener listener); //添加 单个 Future 回调到链表头部void _prependListeners(_FutureListener? listeners) //将回调链添加到链表头部void _removeListeners(); //移除并返回当前Future的所有监听器void __reverseListeners(_FutureListener? listeners); //反转链表}

从我们贴出的 _Future 类结构可以看出,它是 Future 真正的底层实现,承担了 Dart 异步机制中绝大部分实际工作。其主要功能可以拆解为以下四个部分:

  1. 状态管理
    _Future 本质上是一个状态机(state machine)。它管理异步任务的生命周期:未完成 → 完成(成功/失败)→ 回调派发。
    1.1 _state
    _state 描述了可能出现的状态,包括:
  • incomplete:Future 尚未完成;
  • value:Future 已完成,有一个正常结果;
  • error:Future 已完成,抛出了异常;
  • chained:Future 自己不持有结果,而是依赖另一个 Future。
    并允许组合判断状态:
    bool get _isComplete => (_state & (_stateValue | _stateError)) != 0;
    bool get _hasError => (_state & _stateError) != 0;
    1.2 _resultOrListeners
    _resultOrListeners 是 _Future 状态同步的关键容器字段,它的类型是 var,也就是说它可以根据当前状态动态切换存储内容:
  • Future未完成时: 注册的 listener 链表;
  • 已完成,有一个正常结果:完成的值;
  • Future 已完成,抛出了异常:异常对象 (AsyncError);
  • Future 自己不持有结果,而是依赖另一个 Future:链式模式中所依赖的源 _Future。

可以说,_resultOrListeners 是整个 _Future 状态机的“状态内容仓库”,配合 _state 一起完成了状态标志 + 状态数据的组合表达。
2. 链式触发
当我们在一个 Future 上调用 .then()、.catchError() 或 .whenComplete() 时,Dart 会创建一个新的 Future 并注册对应的回调,当原始 Future 完成后触发回调,新的 Future 则根据回调的返回值决定如何完成。如果返回的是普通值,新 Future 立即完成;如果返回的是另一个 Future,则进入链式等待,直到该 Future 完成。

Future<R> then<R>(FutureOr<R> f(T value), {Function? onError}) {//...错误处理_Future<R> result = new _Future<R>();_addListener(new _FutureListener<T, R>.then(result, f, onError));return result;}

梳理一下then方法执行的过程:

  1. 创建一个新的 _Future result;
  2. 构造一个 _FutureListener 对象,把回调绑定进去;
  3. 把这个 listener 挂在当前 Future(source)上;
  4. 当前 Future 完成时,调度执行 listener 的回调;

如果回调返回了另一个 Future(记作 returned),Dart 会调用 _chainCoreFutureSync(returned, result),让 target 依赖于 returned 的完成状态,构建链式触发关系。

链式(Chain)
chain 的意思是:我这个 Future 不自己决定什么时候完成,而是等另一个 Future 完成,把它的结果“搬过来”当作自己的结果。

那么出现了一个问题:为什么 result 明明是等 source 完成才有结果,但 Dart 不把它们 chain,在回调里返回 Future 的时候却要 chain?
我们来深入分析 _propagateToListeners() 中对回调返回值的判断处理,是如何决定当前 result 的完成方式的。
关键代码如下:

var returned = listener.handleValue(source._resultOrListeners);if (returned is Future && !identical(returned, listener.result)) {// 回调返回了一个 Future(而且不是自己)_chainCoreFutureSync(returned, listener.result); // 关键链式触发点
} else {// 回调返回的是普通值,直接完成listener.result._complete(returned);
}

问题就变成了:当source Future执行完成时,如果回调函数的返回值是数值就去执行_complete()方法,结束result Future, 如果返回值类型是Future类型会将 新的Future(returned) 与result Future的生命周期绑定。这两种不都是一个结束另一个也结束吗?

主要问题在于: 情况1 的Future A结束 ,B也结束,并不是同时完成的,换句话说,Future A结束只是通过回调去 通知 B去执行并变成完成状态,而 情况2 是B 完全放弃自己的生命周期管理,将自己的回调节点完全转移到A的回调链上。
其实 情况1 更符合我们认知的chain的概念,但是情况2是Future设计的精髓,await 关键词将修饰的Future与异步函数返回的Future绑定本质是也是通过chain实现。
Chain 建立流程
下面我们看一下chain建立的过程
2.1 监听器转移机制 :

  • 当调用 _setChained 建立链式关系时,会将当前Future的监听器转移到源Future
// 在_chainCoreFutureSync中_FutureListener ? listeners = target._resultOrListeners;target. _setChained ( source ) ;source. _prependListeners ( listeners ) ; // 将监听器转移到源Future

这个过程表示:
target 自己不执行任何回调了,而是“把所有后续操作都挂到 source(returned)上”,由它完成后再向下传播。

2.2 完成传播路径 :
当源 Future(比如某个回调返回的 Future)完成后:

  • 会调用 _propagateToListeners(),遍历并通知其所有监听器;
  • 如果这些监听器来自于一个链式 Future(如上面的 target),它们会被特殊处理:
    • 通过 _FutureListener 的子类包装,确保在值传递的同时更新目标 result 的状态;
    • 例如 .then() 的监听器会调用 handleValue(),并可能再触发 _complete() 或新的 chain。

2.3 状态检查 :
Dart 会通过以下两个方式标识和追踪链式状态:
_isChained

  • 这是 target 的状态位;
  • 当 target._setChained(source) 被调用时会设置;
  • 表示当前 Future 并不自己持有结果,而是依赖另一个 Future;
  • 它的 _resultOrListeners 字段也被设置为 source,表示链的指向。

_chainSource

  • 提供链式结构中当前 Future 所依赖的源 Future;
  • 如果是链式 Future,_chainSource 返回的是 _resultOrListeners 中的源 Future;
  • 可以用于调试、诊断依赖结构,或从链式 Future 中找出最初的结果拥有者。

总结
当 Dart 判断一个 Future 需要被 chain(即回调返回了另一个 Future),它会让当前 Future 的回调挂到 returned Future 上,完成后再统一传播,并通过 _isChained 和 _resultOrListeners 标记整个依赖关系。
3. 异步调度

- _Completer<T> / _AsyncCompleter<T> / _SyncCompleter<T>:提供手动完成 Future 的机制。
- _complete / _asyncComplete :完成Future的方法_Completer<T>及其子类abstract class _Completer<T> implements Completer<T> {final _Future<T> future = _Future<T>();void complete([FutureOr<T>? value]);void completeError(Object error, [StackTrace? stackTrace]);
}class _AsyncCompleter<T> extends _Completer<T> { /* microtask完成Future */ }
class _SyncCompleter<T> extends _Completer<T> { /* 同步立即完成Future */ }

用法区别:

  • Completer() = _AsyncCompleter(异步完成,本质是调用_asyncComplete)
  • Completer.sync() = _SyncCompleter(同步完成,本质是调用_complete)

_complete() & _asyncComplete()
Future 支持“立即完成”与“异步完成”两种模式。这一机制由 _complete() 与 _asyncComplete() 方法控制,调用者可以通过不同方式选择调度路径。

_complete():同步完成

void _complete(FutureOr<T> value) {assert(!_isComplete);if (value is Future<T>) {if (value is _Future<T>) {_chainCoreFutureSync(value, this);} else {_chainForeignFuture(value);}} else {_FutureListener? listeners = _removeListeners();_setValue(value);_propagateToListeners(this, listeners);}}
  • 直接完成当前 Future,立即更新状态并分发监听器;
  • 不进入 microtask 队列,适合需要同步完成的场景;
  • 用于 Completer.sync() 创建的同步 Completer。

_asyncComplete():异步完成

void _asyncComplete(FutureOr<T> value) {assert(!_isComplete); // 保证当前 Future 还没完成if (value is Future<T>) {_chainFuture(value); // 如果是 Future,立即链式绑定return;}_asyncCompleteWithValue(value); // 如果是数值,推入微任务中完成
}
  • 将 _complete() 推入 microtask 队列中异步执行;
  • 保证 Future 的完成不会打断当前同步代码执行栈;
  • 是 Completer() 默认使用的路径。
  1. 回调管理
  2. _FutureListener<S, T>
  • _FutureListener<S, T>:监听回调单元,代表 .then()、.catchError()、.whenComplete() 等注册。
class _FutureListener<S, T> {final _Future<T> result; // 回调结果将赋值到的新 Future。final int state; // 掩码标记当前 listener 类型。final Function? callback; // 正常回调final Function? errorCallback; // 错误回调FutureOr<T> handleValue(S value); // 成功回调执行FutureOr<T> handleError(AsyncError asyncError); // 错误回调执行dynamic handleWhenComplete(); // 完成回调执行
}

state取值如下:

  • stateThen:处理成功值
  • stateCatchError:处理异常
  • stateWhenComplete:处理 finally 场景
  1. Listener 链组织结构:单链表

所有 _FutureListener 是以单向链表 结构 串接的:

  • 每个 _Future 拥有一个 _resultOrListeners 字段;
  • 在未完成状态时,这个字段保存的是 listener 链的头节点;
  • 每次调用 .then() 或 .catchError() 都会创建一个 _FutureListener,并通过 _addListener() 加入链头;
  • 执行完成后,listener 会被 _removeListeners() 清空。
  1. 操作方法
  2. _addListener(_FutureListener listener)
  • 将 listener 添加到链表头部;
  • 如果当前 Future 已完成,则立即派发执行;
  1. _prependListeners(_FutureListener? listeners)
  • 将一整段 listener 链表接到当前 listener 链的头部;
  • 用于链式传播中 listener 的转移,例如在 _chainCoreFutureSync() 中:
final listeners = target._removeListeners();
target._setChained(source);
source._prependListeners(listeners);
  1. _removeListeners()
  • 将当前 Future 所有 listener 移除并返回(用于调度);
  • 移除后 _resultOrListeners 置为 null 或结果值;
  1. __reverseListeners(_FutureListener? listeners)
  • 于反转链表顺序;
  • 保证按注册顺序依次触发,而不是后注册先触发;
  1. 完整流程
  2. Future 调用 .then() 注册回调,创建 _FutureListener,加入链表;
  3. Future 完成后,调用 _propagateToListeners():
  • 遍历 listener 链;
  • 区分状态执行 handleValue / handleError / handleWhenComplete;
  • 将回调返回结果赋值到 listener.result;
  1. 若返回的是 Future,则触发 _chainCoreFutureSync(),建立链式;

  2. _propagateToListeners

static void _propagateToListeners(_Future source, _FutureListener? listeners) {while (true) {if (listeners == null) {if (source._hasError && !source._ignoreError) {source._zone.handleUncaughtError(source._error);}return;}final listener = listeners;listeners = listener._nextListener;// 执行handleXXX回调if (listener.handlesComplete) {listener.handleWhenComplete();} else if (!source._hasError) {listener.handleValue(source._resultOrListeners);} else {listener.handleError(source._error);}// 链式传播或继续下一个 listener}
}

功能:

  • 调用合适的回调(handleValue/handleError/handleWhenComplete)
  • 把回调结果继续传播给下一个 Future
  1. _chainCoreFutureSync
static void _chainCoreFutureSync(_Future source, _Future target) {if (source._isComplete) {target._cloneResult(source);_propagateToListeners(target, target._removeListeners());} else {target._setChained(source);source._prependListeners(target._removeListeners());}
}

功能:

  • 如果 source Future 已完成,直接拷贝它的结果。
  • 如果 source Future 未完成,把 target 链到 source 上等待其完成。
  1. 错误信息管理
  • AsyncError:统一封装错误与堆栈信息。
  • Zone:隔离执行上下文,管理回调与未捕获异常。
    2.4 AsyncError
class AsyncError {final Object error;final StackTrace stackTrace;
}

标准错误封装,用于 Future 错误状态记录与传播。

2.5 Zone相关

  • 所有 Future 回调 (then/catchError/whenComplete) 都在注册时的 Zone 执行。
  • Future 完成时如果未捕获错误,通过 Zone 的 handleUncaughtError 抛出。

相关文章:

Flutter异步原理-Future

前言 在 Dart 中&#xff0c;谈到异步就离不开 Future。无论是 .then()、还是 await&#xff0c;它们背后运作的都是一个私有实现类&#xff1a;_Future &#xff0c;我们平时使用的 Future 只是一个抽象接口&#xff0c;其真正的实现逻辑由_Future 承担。 class _Future<…...

TRAE 配置blender MCP AI自动3D建模

BlenderMCP - Blender模型上下文协议集成 BlenderMCP通过模型上下文协议(MCP)将Blender连接到Claude AI&#xff0c;允许Claude直接与Blender交互并控制Blender。这种集成实现了即时辅助的3D建模、场景创建和操纵。 1.第一步下载 MCP插件(addon.py):Blender插件&#xff0c;在…...

VUE2课程计划表练习

主要练习数据变量对象 以下是修正后的完整代码&#xff1a; //javascript export default {data() {return {list: [{ id: 1, subject: Vue.js 前端实战开发, content: 学习指令&#xff0c;例如 v-if、v-for、v-model 等, place: 自习室, status: false }// 可以在这里添加更…...

虚拟文件系统

虚拟文件系统&#xff08;Virtual File System&#xff0c;VFS&#xff09;是操作系统内核中的一个抽象层&#xff0c;它为不同的文件系统&#xff08;如ext4、NTFS、FAT32等&#xff09;提供统一的访问接口。通过VFS&#xff0c;用户和应用程序无需关心底层文件系统的具体差异…...

2025年软件工程与数据挖掘国际会议(SEDM 2025)

2025 International Conference on Software Engineering and Data Mining 一、大会信息 会议简称&#xff1a;SEDM 2025 大会地点&#xff1a;中国太原 收录检索&#xff1a;提交Ei Compendex,CPCI,CNKI,Google Scholar等 二、会议简介 2025年软件开发与数据挖掘国际会议于…...

基于大模型预测的足月胎膜早破行阴道分娩全流程研究报告

目录 一、引言 1.1 研究背景与意义 1.2 研究目的与方法 1.3 研究创新点 二、胎膜早破(足月)行阴道分娩概述 2.1 胎膜早破定义与分类 2.2 足月胎膜早破行阴道分娩的现状与挑战 2.3 大模型预测引入的必要性 三、大模型预测原理与技术 3.1 大模型介绍 3.2 数据收集与…...

学习记录:DAY28

DispatcherController 功能完善与接口文档编写 前言 没什么动力说废话了。 今天来完善 DispatcherController 的功能&#xff0c;然后写写接口文档。 日程 早上&#xff1a;本来只有早八&#xff0c;但是早上摸鱼了&#xff0c;罪过罪过。下午&#xff1a;把 DispatcherContro…...

软件系统中功能模型 vs 数据模型 对比解析

功能模型 vs 数据模型 对比解析 一、功能模型&#xff08;Functional Model&#xff09; 定义&#xff1a;描述系统 做什么&#xff08;业务逻辑与操作流程&#xff09; 核心关注&#xff1a;行为、交互、业务流程 建模工具&#xff1a; 用例图&#xff08;UML Use Case Dia…...

.NET高频技术点(持续更新中)

1. .NET 框架概述 .NET 框架的发展历程.NET Core 与 .NET Framework 的区别.NET 5 及后续版本的统一平台 2. C# 语言特性 异步编程&#xff08;async/await&#xff09;LINQ&#xff08;Language Integrated Query&#xff09;泛型与集合委托与事件属性与索引器 3. ASP.NET…...

pandas中的数据聚合函数:`pivot_table` 和 `groupby`有啥不同?

pivot_table 和 groupby 是 pandas 中两种常用的数据聚合方法&#xff0c;它们都能实现数据分组和汇总&#xff0c;但在使用方式和输出结构上有显著区别。 0. 基本介绍 groupby分组聚合 groupby 是 Pandas 库中的一个功能强大的方法&#xff0c;用于根据一个或多个列对数据进…...

微调大模型如何准备数据集——常用数据集,Alpaca和ShareGPT

微调大模型如何准备数据集——常用数据集,Alpaca和ShareGPT 数据集准备常用数据集自定义数据集AlpacaShareGPT数据集准备 常用数据集 预训练数据集 Wiki Demo (en)RefinedWeb (en)RedPajama V2 (en)Wikipedia (en)Wikipedia (zh)Pile (en)...

【Gradio】helloworld程序

前言 发现这个库用来做可视化的demo还不错&#xff0c;简单学习一下。 官网 https://www.gradio.app/ 安装 pip install gradio -i https://pypi.tuna.tsinghua.edu.cn/simple/helloWorld 示例 import gradio as grdef greet(name):return "hello"nameifacegr…...

机器学习例题——预测facebook签到位置(K近邻算法)和葡萄酒质量预测(线性回归)

一、预测facebook签到位置 代码展示&#xff1a; import pandas as pd from sklearn.preprocessing import StandardScaler from sklearn.model_selection import train_test_split from sklearn.neighbors import KNeighborsClassifier from sklearn.model_selection import…...

对golang中CSP的理解

概念&#xff1a; CSP模型&#xff0c;即通信顺序进程模型&#xff0c;是由英国计算机科学家C.A.R. Hoare于1978年提出的。该模型强调进程之间通过通道&#xff08;channel&#xff09;进行通信&#xff0c;并通过消息传递来协调并发执行的进程。CSP模型的核心思想是“不要通过…...

使用 pgrep 杀掉所有指定进程

使用 pgrep 杀掉所有指定进程 pgrep 是一个查找进程 ID 的工具&#xff0c;结合 pkill 或 kill 命令可以方便地终止指定进程。以下是几种方法&#xff1a; 方法1&#xff1a;使用 pkill&#xff08;最简单&#xff09; pkill 进程名例如杀掉所有名为 “firefox” 的进程&…...

Missashe考研日记-day36(改版说明)

Missashe考研日记-day36 改版说明 经过一天的思考、纠结和尝试&#xff0c;博主决定对更新内容进行改版&#xff0c;如下&#xff1a;1.不再每天都发一篇日记&#xff0c;改为一周发一篇包含一周七天学习进度的周记&#xff0c;但为了标题和以前相同&#xff08;强迫症&#…...

基于Jetson Nano与PyTorch的无人机实时目标跟踪系统搭建指南

引言&#xff1a;边缘计算赋能智能监控 在AIoT时代&#xff0c;将深度学习模型部署到嵌入式设备已成为行业刚需。本文将手把手指导读者在NVIDIA Jetson Nano&#xff08;4GB版本&#xff09;开发板上&#xff0c;构建基于YOLOv5SORT算法的实时目标跟踪系统&#xff0c;集成无人…...

【LunarVim】CMake LSP配置

在 LunarVim 中为 CMakeLists.txt 文件启用代码提示&#xff08;如补全和语义高亮&#xff09;&#xff0c;需要安装支持 CMake 的 LSP&#xff08;语言服务器&#xff09;和适当的插件。以下是完整配置指南&#xff1a; 1、配置流程 1.1 安装cmake-language-server 通过 Ma…...

Mkdocs页面如何嵌入PDF

嵌入PDF 嵌入PDF代码 &#xff0c;注意PDF的相对地址 <iframe src"../个人简历.pdf (相对地址)" width"100%" height"800px" style"border: 1px solid #ccc; overflow: auto;"></iframe>我的完整代码&#xff1a; <d…...

从零开始学Flink:开启实时计算的魔法之旅

在凌晨三点的数据监控大屏前&#xff0c;某电商平台的技术负责人突然发现一个异常波动&#xff1a;支付成功率骤降15%。传统的数据仓库此时还在沉睡&#xff0c;而基于Flink搭建的实时风控系统早已捕捉到这个信号&#xff0c;自动触发预警机制。当运维团队赶到时&#xff0c;系…...

融合静态图与动态智能:重构下一代智能系统架构

引言&#xff1a;智能系统的分裂 当前的大模型系统架构正处于两个极端之间&#xff1a; 动态智能体系统&#xff1a;依赖语言模型动态决策、自由组合任务&#xff0c;智能灵活但稳定性差&#xff1b; 静态流程图系统&#xff1a;具备强工程能力&#xff0c;可控可靠&#xf…...

滑动窗口-窗口中的最大/小值-单调队列

求窗口的最大值 #include <iostream> //滑动窗口最大值用单调队列q[]&#xff0c;q存储候选最大值的下标 //队列头是最大值的下标 using namespace std; const int N100010; int nums[N],q[N]; int hh0,tt-1;// hh 是队头指针&#xff0c;tt 是队尾指针&#xff0c;初始…...

Java设计模式全面详解:从基础到高级的23种模式简介

引言:设计模式概述 设计模式是软件设计中常见问题的典型解决方案,它们就像预制的蓝图,你可以通过调整来解决代码中反复出现的设计问题。Java设计模式通常分为三大类: 创建型模式:处理对象创建机制结构型模式:处理对象组合行为型模式:处理对象间的交互与职责分配#mermai…...

WORD压缩两个免费方法

日常办公和学习中&#xff0c;Word文档常常因为包含大量图片、图表或复杂格式而导致文件体积过大&#xff0c;带来诸多不便&#xff0c;比如 邮件发送受限&#xff1a;许多邮箱附件限制在10-25MB&#xff0c;大文件无法直接发送 存储空间占用&#xff1a;大量文档占用硬盘或云…...

skywalking服务安装与启动

skywalking服务安装并启动 1、介绍2、下载apache-skywalking-apm3、解压缩文件4、创建数据库及用户5、修改配置文件6、下载 MySQL JDBC 驱动7、启动 OAP Serve,需要jkd11,需指定jkd版本,可以修改文件oapService.sh8、启动 Web UI,需要jkd11,需指定jkd版本,可以修改文件oapServi…...

Qt 中信号与槽(signal-slot)机制支持 多种连接方式(ConnectionType)

Qt 中信号与槽&#xff08;signal-slot&#xff09;机制支持 多种连接方式&#xff08;ConnectionType&#xff09; Qt 中信号与槽&#xff08;signal-slot&#xff09;机制支持 多种连接方式&#xff08;ConnectionType&#xff09;&#xff0c;用于控制信号发出后如何调用槽…...

Booth Encoding vs. Non-Booth Multipliers —— 穿透 DC 架构看乘法器的底层博弈

目录 &#x1f9ed; 前言 &#x1f331; 1. Non-Booth 乘法器的实现原理&#xff08;也叫常规乘法器&#xff09; &#x1f527; 构建方式 ✍️ 例子&#xff1a;4x4 Non-Booth 乘法器示意 &#x1f9f1; 硬件结构 ✅ 特点总结 ⚡ 2. Booth Encoding&#xff08;布斯编码…...

Midjourney-V7:支持参考图片头像或背景生成新保真图

Midjourney-V7重磅升级Omni Reference&#xff1a;全能图像参考神器&#xff01;再也不用担心生成图片货不对版了&#xff01; 就在上周&#xff0c;Midjourney发版它最新的V7版本&#xff1a;Omini Reference&#xff0c;提供了全方位图像参考功能&#xff0c;它可以参考你提…...

耀圣-气动带刮刀硬密封法兰球阀:攻克颗粒高粘度介质的自清洁 “利器”

气动带刮刀硬密封法兰球阀&#xff1a;攻克颗粒高粘度介质的自清洁 “利器” 在化工、矿业、食品加工等行业中&#xff0c;带颗粒高粘度介质、料浆及高腐蚀性介质的输送与控制一直是行业难题。普通阀门极易因介质附着、颗粒堆积导致卡阻失效&#xff0c;密封面磨损加剧&#x…...

Google云计算原理和应用之分布式锁服务Chubby

Chubby是Google设计的提供粗粒度锁服务的一个文件系统,它基于松耦合分布式系统,解决了分布的一致性问题。通过使用Chubby的锁服务,用户可以确保数据操作过程中的一致性。不过值得注意的是,这种锁只是一种建议性的锁(Advisory Lock)而不是强制性的锁,这种选择系统具有更大…...