Flutter中 yield*关键字

yield* 是 Dart 语言(Flutter 使用 Dart)中的一个关键字,用于生成器函数 中,表示将另一个生成器中的所有值逐个产出

一、yield* 的基本概念

生成器函数是可以多次返回值 的函数,使用 sync*(同步)或 async*(异步)标记。

复制代码
// 同步生成器:返回 Iterable
Iterable<int> syncGenerator() sync* {
  yield 1;
  yield 2;
  yield 3;
}

// 异步生成器:返回 Stream
Stream<int> asyncGenerator() async* {
  yield 1;
  yield 2;
  yield 3;
}

2. yield vs yield*

关键字 作用 产出
yield 产出一个值 单个值
yield* 产出另一个生成器的所有值 多个值
复制代码
// 使用 yield
Iterable<int> withYield() sync* {
  yield 1;
  yield 2;
  yield 3;
}
// 产出:1, 2, 3

// 使用 yield*
Iterable<int> withYieldStar() sync* {
  yield* [1, 2, 3];  // 展开列表中的所有值
}
// 产出:1, 2, 3(效果相同)

二、yield* 的语法和用法

1. 基本语法

复制代码
// 同步生成器中使用 yield*
Iterable<T> generator() sync* {
  yield* otherIterable;  // 产出 otherIterable 中的所有元素
}

// 异步生成器中使用 yield*
Stream<T> asyncGenerator() async* {
  yield* otherStream;    // 产出 otherStream 中的所有事件
}

2. 实际示例

复制代码
// 示例 1:展开列表
Iterable<int> flattenList() sync* {
  yield* [1, 2, 3];
  yield* [4, 5, 6];
}

void main() {
  print(flattenList().toList());  // [1, 2, 3, 4, 5, 6]
}

// 合并多个数据源
Stream<int> mergeStreams() async* {
  yield* Stream.fromIterable([1, 2, 3]);
  yield* Stream.fromIterable([4, 5, 6]);
  yield* Stream.fromIterable([7, 8, 9]);
}

// 使用
StreamSubscription? subscription;
subscription = mergeStreams().listen((value) {
  print(value);  // 1,2,3,4,5,6,7,8,9
});

四、yield* vs 其他方式

1. yield* vs for 循环

复制代码
// 使用 yield*(简洁)
Iterable<int> withYieldStar() sync* {
  yield* [1, 2, 3];
}

// 使用 for 循环(冗长)
Iterable<int> withForLoop() sync* {
  for (var item in [1, 2, 3]) {
    yield item;
  }
}

// 效果完全相同,但 yield* 更简洁

2. yield* vs await for

复制代码
// 异步生成器中使用 yield*
Stream<int> withYieldStar() async* {
  yield* Stream.fromIterable([1, 2, 3]);
}

// 使用 await for(等效,但更冗长)
Stream<int> withAwaitFor() async* {
  await for (var value in Stream.fromIterable([1, 2, 3])) {
    yield value;
  }
}

五、注意事项

1. 不能混用同步和异步

复制代码
// 错误:同步生成器中不能使用异步源
Iterable<int> wrong() sync* {
  yield* Stream.fromIterable([1, 2, 3]);  // 编译错误!
}

// 正确:同步生成器只能用于同步源
Iterable<int> correct() sync* {
  yield* [1, 2, 3];  // 正确
}

2. 类型必须匹配

复制代码
// 错误:类型不匹配
Stream<String> wrong() async* {
  yield* Stream.fromIterable([1, 2, 3]);  // 编译错误!类型不匹配
}

// 正确:类型必须一致
Stream<int> correct() async* {
  yield* Stream.fromIterable([1, 2, 3]);  // 正确
}

六、对比表

特性 yield yield*
产出数量 单个值 多个值(展开另一个生成器)
使用场景 产出一个值 委托给另一个生成器
递归支持 需要手动递归 天然支持递归
性能 直接产出 直接委托,无开销
代码简洁度 一般 高(避免嵌套循环)

备注

  1. yield* 用于将另一个生成器的所有值逐个产出

  2. 常用于递归遍历Stream 组合分页加载

  3. 比手动 for 循环或 await for简洁高效

  4. 必须在同类型生成器 中使用(sync*sync*async*async*

相关推荐
程序员老刘16 小时前
跨平台开发地图 | 2026年6月
flutter·ai编程·客户端
悟空瞎说1 天前
Flutter 架构详解:新手必懂底层原理
flutter
SoaringHeart2 天前
Flutter最佳实践:IM聊天文字链接自动识别跳转
前端·flutter
恋猫de小郭2 天前
KMP / CMP 鸿蒙版本 Beta 发布,他有什么特别之处?
android·前端·flutter
风华圆舞3 天前
Flutter + 鸿蒙 Intents Kit:页面直达能力的完整接入方案
flutter·ui·华为·harmonyos
韩曙亮3 天前
【Flutter】Flutter 组件 ④ ( 组件渲染 的 三棵树理论 | Widget 树 → Element 树 → RenderObject 树 )
flutter·element·widget·renderobject
恋猫de小郭3 天前
Android 17 正式版发布,全新 AI 和各种破坏性更新
android·前端·flutter
kingbal3 天前
Windows:flutter环境搭建
windows·flutter
911hzh3 天前
Flutter MethodChannel 跨端通信框架 zh_native_channel:快速入门、优势分析与 Pigeon 对比
flutter
911hzh3 天前
Flutter 快速搭建新项目:用 Flutter Foundation Kit 一条命令生成带基础架构的 App 模板
flutter