Flutter中的Stream初探

Stream: 超级抽象的一个XXX

Stream 的分类

(1) "Single-subscription" -- 单订阅流
(2) "broadcast" -- 广播式的流(可多订阅)

  • 单订阅流只能被订阅一次,重复订阅会报错, 直到设置listen 后才会发送。单订阅流通常用于流式数据块较大的连续数据,如文件I/O。
  • 广播式的 可以订阅多次,在listen之前的数据会丢失。
    - 如果一个流是单订阅模式 却想多次订阅,可以通过asBroadcastStream ()方法来修改。

Stream 的创建

(1) 从集合中创建一个新的单订阅流, Stream.fromIterable
Stream stream1 = Stream.fromIterable([11, 22, 33]);
(2) 从Future中创建一个新的单订阅流, Stream.fromFuture
Stream stream = Stream.fromFuture(Future(()=> 1));
(3) 通过Stream.fromFutures创建
Stream stream3 = Stream.fromFutures([
Future(() => 111),
Future(() => 111),
]);
(4) 创建一个每隔自定义时间发送一个数据的流
Stream stream2 = Stream.periodic(Duration(seconds: 2), (a) {
print("a------>$a");
if (a < 3)
return a;
else
return a *10;
});
它有2 个参数, 第一个是间隔的时间, 第二个是每次发送数据前的回调方法,可以通过这个方法的返回值来修改流的值。 (返回值即向流中添加的内容)
上面创建的流是一个固定间隔时间无限发送的流,这就有问题了。 正常情况下谁会去搞一个无限发送数据流的功能呢? 怎样控制它的结束?-----------
用stream的take 方法。 stream2 = stream2.take(5); 这样就只会发送5次了。
升级版的takeWhile () : 可以用来做筛选

stream2 = stream2.takeWhile((data) {
 return data < 10; // 设置一个上限为10 
});
image.png

为了方操作 Stream ,官方提供了StreamController;如上图所示,他提供StreamSink来添加流 (入口),同时又提供 stream 属性用于对外的监听和变换。 stream.listen的返回时一个StreamSubscription,可以通过它的pause(),resume(),cancel()等方法来操作流的订阅。

StreamController:
StreamController controller = StreamController<String>(); // 创建一个单订阅流
StreamController controller = StreamController.broadcast(); // 创建一个广播式的订阅流
参数sync用来指定是同步还是异步。

listen : 用来设置监听, 它的返回值是 StreamSubscribe。
StreamSubscribe:
pause() : 暂停监听(是立即暂停),暂停后的事件流不会丢失,会在resume后一起回调
resume(): 唤醒pause的流
cancel(): 取消
举个栗子🌰

 // 1. StreamControl
StreamController controller = StreamController<String>();
// 2. StreamSink
StreamSink sink = controller.sink;
// 3. Stream
Stream stream = controller.stream;
stream.transform(StreamTransformer<String, String>.fromHandlers(
    handleData: (String data, EventSink<String> sink) {
  // 在这里设置transform 是没有用的,不会走这里; 除非在stream.transform返回的stream上加listen监听。
  if (!data.contains("数据2")) {
    sink.add(data);
  }
}));
sink.add("3秒后才设置监听。");
// 4. subscribe
Timer(Duration(seconds: 3), () {
  StreamSubscription subscription = controller.stream.transform(
      StreamTransformer<String, String>.fromHandlers(
          handleData: (String data, EventSink<String> sink) {
    print("transform");
    if (!data.contains("数据3")) {
      sink.add(data);
    }
  })).listen((event) {
    print("接收到新的消息: " + event);
  });
  sink.add("我是一条新的数据"); 
  Timer(Duration(milliseconds: 100), () {
    sink.add("pause...");
    subscription.pause(); // 暂停
    sink.add("我是一条新的数据pause"); 
  });

  Timer(Duration(seconds: 5), () {
    subscription.resume();

    sink.add("我是一条新的数据2"); 
  });
});

输出结果: 绿色的先输出,过5秒后黄色的才输出


image1.png

那么问题来了, 为什么pause 之前的add的那一个流没有输出呢? 跟进去看源码就明白了,pause 期间是不会分发事件的。


image2.png

schedule 的实现
image3.png

这样也就明白了, Stream最终是 想microtask queue 中添加了一个microtask 来实现异步的功能。

说点题外的: flutter是单线程,他的异步实现是通过Event Looper 来实现的。Event looper 中包含2个队列: (1)MicorTask Queue (2) Event Queue , MicroTask 的优先级是大于Event Queue的,只有所有的MicroTask Queue中的任务都完成以后才会去执行Event Queue中的内容。

当然 StreamController 可以是同步的,只要在创建的时候将参数sync设置为true即可,sync: true

如何通过Stream来实现响应式的组件 ?

通过StreamBuilder

看个例子:

class StreamModel {
  StreamController _controller;

  StreamSink<List<BookResponseData>> _sink;

  Stream<List<BookResponseData>> stream;

  StreamModel() {
    // 构造方法中初始化流相关的对象
    _controller = StreamController<List<BookResponseData>>.broadcast();
    _sink = _controller.sink;
    stream = _controller.stream;
  }
  /// 获取书本列表
  getBookList() async {
    var httpClient = new HttpClient();
    var uri = new Uri.https('www.apiopen.top', '/novelApi');
    var request = await httpClient.getUrl(uri);
    var response = await request.close();
    var responseBody = await response.transform(utf8.decoder).join();
    // 将获取到的字符串转换成定义好的Book实体类
    BookResponseEntity entity =
        BookResponseEntity.fromJson(json.decode(responseBody));
    // 接口中拿到数据之后,通过sink.add 添加一条流即可, 这样在StreamBuild中就会有回调。
    _sink.add(entity.data);
  }
  /// 资源
  dispose() {
    _sink.close();
    _controller.close();
  }
}

组件类:

class BookList extends StatefulWidget {
  @override
  State<StatefulWidget> createState() {
    return _BookListState();
  }
}

//https://api.apiopen.top/getSingleJoke?sid=28654780
//https://www.apiopen.top/novelApi

class _BookListState extends State<BookList> {
  StreamModel streamModel;

  @override
  void initState() {
    super.initState();
    streamModel = StreamModel();
  }

  @override
  Widget build(BuildContext context) {
//    print("build");
    return Scaffold(
      appBar: AppBar(
        title: Text("stream demo"),
      ),
      body: Container(
        child: StreamBuilder<List<BookResponseData>>(
          stream: streamModel.stream,  // 要监听的流 
          initialData: [], // 初始值,可以不设
          builder: (context, a) { // sink.add 后,就会回调这个方法。
            List<Widget> views = [];
            if (a.data != null && a.data.length > 0) {
              a.data.forEach((BookResponseData data) {
                views.add(Container(
                  padding: EdgeInsets.all(10.0),
                  child: Column(
                    children: <Widget>[
                      Text(
                        data.bookname,
                        style: TextStyle(fontWeight: FontWeight.w600),
                      ),
                      Text(data.bookInfo),
                    ],
                  ),
                ));
              });
            }
            return ListView(
              children: views,
            );
          },
        ),
      ),
      floatingActionButton: FloatingActionButton(
          child: Text("获取数据"),
          onPressed: () {
            streamModel.getBookList();
          }),
    );
  }
  @override
  void dispose() {
    streamModel.dispose(); 
    super.dispose();
  }
}

以上便可以实现基于Stream的响应式组件。

又有问题了。。。。 没办法问题就是这么多 (来打我... )

为什么StreamBuilder能够监听到Stream的变化来刷新UI?
跟进去看一下源码

void _subscribe() {
  if (widget.stream != null) {
    _subscription = widget.stream.listen((T data) {
      setState(() {
        _summary = widget.afterData(_summary, data);
      });
    }, onError: (Object error) {
      setState(() {
        _summary = widget.afterError(_summary, error);
      });
    }, onDone: () {
      setState(() {
        _summary = widget.afterDone(_summary);
      });
    });
    _summary = widget.afterConnected(_summary);
  }
}

你会发现 StreamBuilder是一个StatefulWidget, 本质还是在stream.listen中通过setState 来实现响应数据刷新View。

这样,对Stream及基于Stream的响应式组件就有个大致的了解了... ...

over。。。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,185评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,445评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,684评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,564评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,681评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,874评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,025评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,761评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,217评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,545评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,694评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,351评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,988评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,778评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,007评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,427评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,580评论 2 349

推荐阅读更多精彩内容