Flutter异步原理-Future

news2025/5/10 22:15:02

前言

  • 在 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); // 异步链接 Future
  void _chainForeignFuture(Future source); // 链接外部 Future
  
  void _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 抛出。

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

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

相关文章

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 }// 可以在这里添加更…

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

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

.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;用于根据一个或多个列对数据进…

对golang中CSP的理解

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

【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…

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

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

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;用于控制信号发出后如何调用槽…

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)而不是强制性的锁,这种选择系统具有更大…

SM2Utils NoSuchMethodError: org.bouncycastle.math.ec.ECFieldElement$Fp.<init

1&#xff0c;报错图示 2&#xff0c;报错原因&#xff1a; NoSuchMethodError 表示运行时找不到某个方法&#xff0c;通常是编译时依赖的库版本与运行时使用的库版本不一致。 错误中的 ECFieldElement$Fp. 构造函数参数为 (BigInteger, BigInteger)&#xff0c;说明代码期望使…

《100天精通Python——基础篇 2025 第16天:异常处理与调试机制详解》

目录 一、认识异常1.1 为什么要使用异常处理机制?1.2 语法错误1.3 异常错误1.4 如何解读错误信息 二、异常处理2.1 异常的捕获2.2 Python内置异常2.3 捕获多个异常2.4 raise语句与as子句2.5 使用traceback查看异常2.6 try…except…else语句2.7 try…except…finally语句--捕获…

动态创建链表(头插法、尾插法)

今天我们来学习动态创建链表&#xff01;&#xff01;&#xff01; 动态创建链表&#xff1a;分为头插法和尾插法 头插法&#xff08;动态创建&#xff09;&#xff1a; 头插法就是让新节点变成头 代码如下 吐血了&#xff1a;这边有个非常重要的知识点&#xff0c;这边第三…

利用混合磁共振成像 - 显微镜纤维束成像技术描绘结构连接组|文献速递-深度学习医疗AI最新文献

Title 题目 Imaging the structural connectome with hybrid MRI-microscopy tractography 利用混合磁共振成像 - 显微镜纤维束成像技术描绘结构连接组 01 文献速递介绍 通过多种模态绘制大脑结构能够增进我们对大脑功能、发育、衰老以及疾病的理解&#xff08;汉森等人&am…

安全监控之Linux核心资产SSH连接监测邮件

文章目录 一、引言二、邮箱设置三、脚本配置四、登录测试 一、引言 在某些特殊时期&#xff08;如HVV&#xff09;需要重点监控Linux核心资产SSH连接登录活动情况&#xff0c;实现ssh登录报警监控。其实实现方式并不难。 二、邮箱设置 在邮箱中需要启用“SMTP”协议&#xf…