Flutter 之 Stream的生成

Stream的生成

1拾给、從零開始創(chuàng)建Stream
創(chuàng)建一個Stream可以通過異步生成器(async*)函數(shù)。當異步生成器函數(shù)被調用時會創(chuàng)建一個 Stream兔沃,而函數(shù)體則會在該 Stream 被監(jiān)聽時開始運行蒋得。當函數(shù)返回時,Stream 關閉乒疏。在函數(shù)返回前额衙,你可以使用 yield 或 yield 語句向該 Stream 提交事件。
下面是一個周期性發(fā)送整數(shù)的函數(shù)例子:

void main() {
  var duration = Duration(seconds: 3);
  var stream = timedCounter(duration, 10);
  stream.listen((event) {
    print(event);
  });
}

Stream<int> timedCounter(Duration interval, [int maxCount]) async* {
  var i = 0;
  while(true) {
    await Future.delayed(interval);
    yield i++;
    if(i == maxCount) break;
  }
}

2怕吴、轉換現(xiàn)有的Stream
我們在創(chuàng)建 Stream 時常見的情形是根據(jù)現(xiàn)有 Stream 的事件創(chuàng)建一個新的 Stream窍侧。比如你已經有了一個可以提供字節(jié)事件的 Stream,然后你想將該 Stream 變?yōu)橐粋€可以提供字符串的 Stream转绷,并且該 Stream 中的字符串還經過 UTF-8 編碼伟件。對于這種情況,常用的辦法是創(chuàng)建一個新的 Stream 去等待獲取原 Stream 的事件暇咆,然后再將新 Stream 中的事件輸出锋爪。例如:

/// 將連續(xù)的字符串 Stream 拆分為行。
///
/// 輸入的字符串來自于"源" Stream 并以較小的 chunk 塊提供爸业。
Stream<String> lines(Stream<String> source) async* {
  // 存儲從上一個數(shù)據(jù)塊中分離出的字符串行其骄。
  var partial = '';
  // 等到新的數(shù)據(jù)塊可用時開始處理。
  await for (var chunk in source) {
    var lines = chunk.split('\n');
    lines[0] = partial + lines[0]; // 追加拼接行扯旷。
    partial = lines.removeLast(); // 刪除剩余不完整的行拯爽。
    for (var line in lines) {
      yield line; // 將分離的每個字符串行添加至輸出 Stream。
    }
  }
  // 最后如果最終的字符串行不為空則將其添加至輸出流钧忽。
  if (partial.isNotEmpty) yield partial;
}

3毯炮、使用 StreamController
如果你 Stream 的事件不僅來自于異步函數(shù)可以遍歷的 Stream 和 Future,還來自于你程序的不同部分耸黑,這種情況使用上述兩種方式生成 Stream 就顯得比較困難桃煎。面對這種情況,我們可以使用一個 StreamController 來創(chuàng)建和填充 Stream大刊。
StreamController 可以為你生成一個 Stream为迈,并提供在任何時候、任何地方將事件添加到該 Stream 的方法缺菌。該 Stream 具有處理監(jiān)聽器和暫停所需的所有邏輯葫辐。控制器對象你可以自行處理而只需返回調用者所需的 Stream 即可伴郁。

void main() {
  var duration = Duration(seconds: 3);
  var stream = timedCounter(duration, 10);
  stream.listen((event) {
    print(event);
  });
}

Stream<int> timedCounter(Duration interval, [int maxCount]) {
  var controller = StreamController<int>();
  var counter = 0;
  Timer timer;
  void tick(Timer timer) {
    counter++;
    controller.add(counter); // 請求 Stream 將計數(shù)器值作為事件發(fā)送耿战。
    if (maxCount != null && counter >= maxCount) {
      timer.cancel();
      controller.close(); // 請求 Stream 關閉并告知監(jiān)聽器。
    }
  }

  void startTimer() {
    timer = Timer.periodic(interval, tick);
  }

  void stopTimer() {
    if(timer != null) {
      timer.cancel();
      timer = null;
    }
  }

  controller = StreamController<int> (
    onListen: startTimer,
    onPause: stopTimer,
    onResume: startTimer,
    onCancel: stopTimer,
  );

  return controller.stream;
}

不通過async*函數(shù)創(chuàng)建Stream時焊傅,請務必牢記以下幾點:

  • 使用同步控制器時要小心剂陡。例如狈涮,使用 StreamController(sync: true) 構造方法創(chuàng)建控制器。當你發(fā)送一個事件到一個未暫停的同步控制器(例如:使用 EventSink 中定義的 add()鸭栖、addError()close() 方法)薯嗤,事件立即發(fā)送給所有 Stream 的監(jiān)聽器。在添加監(jiān)聽器的代碼返回之前纤泵,決不能調用 Stream 監(jiān)聽器,而在錯誤的事件使用同步控制器會破壞該規(guī)則并導致其它正常代碼執(zhí)行失敗镜粤。因此捏题,你應該避免使用同步控制器。

  • 如果你使用 StreamController肉渴, onListen 回調會在 listen 方法調用返回 StreamSubscription 前返回公荧。不要讓 onListen 回調依賴于已經存在的訂閱。例如同规,在下面的代碼中循狰,onListen 回調有可能會在 subscription 變量被初始化為一個有效值之前被觸發(fā)(同時 處理器 被調用)

  • 當 Stream 的監(jiān)聽器狀態(tài)改變時,由 StreamController 定義的 onListen券勺、onPause绪钥、onResume 和 onCancel 回調會被調用,該調用絕不會發(fā)生在事件生成時或在某個狀態(tài)變化處理回調的調用期間关炼。在這些情況出現(xiàn)時程腹,狀態(tài)變化的回調會被延遲,直到上一個回調執(zhí)行完成儒拂。

  • 不要嘗試自己去實現(xiàn) Stream 接口寸潦。否則很容易在事件、回調以及添加和移除監(jiān)聽器這些操作交互時出現(xiàn)一些難以察覺的錯誤社痛。你應該總是使用一個現(xiàn)有的 Stream(比如由 StreamController 生成的)去實現(xiàn)新 Stream 中 listen 方法的調用见转。

  • 盡管你可以通過擴展 Stream 類并實現(xiàn) listen 方法來實現(xiàn)更多額外的功能,但一般不建議這么做蒜哀,因為這樣會引入一個調用者必須考慮的新類型斩箫。相反,你可以創(chuàng)建一個(或多個)具有 Stream 的類而不是一個(或多個)Stream凡怎。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末校焦,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子统倒,更是在濱河造成了極大的恐慌寨典,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,723評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件房匆,死亡現(xiàn)場離奇詭異耸成,居然都是意外死亡报亩,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,485評論 2 382
  • 文/潘曉璐 我一進店門井氢,熙熙樓的掌柜王于貴愁眉苦臉地迎上來弦追,“玉大人,你說我怎么就攤上這事花竞【⒓” “怎么了?”我有些...
    開封第一講書人閱讀 152,998評論 0 344
  • 文/不壞的土叔 我叫張陵约急,是天一觀的道長零远。 經常有香客問我,道長厌蔽,這世上最難降的妖魔是什么牵辣? 我笑而不...
    開封第一講書人閱讀 55,323評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮奴饮,結果婚禮上纬向,老公的妹妹穿的比我還像新娘。我一直安慰自己戴卜,他們只是感情好逾条,可當我...
    茶點故事閱讀 64,355評論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著投剥,像睡著了一般膳帕。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上薇缅,一...
    開封第一講書人閱讀 49,079評論 1 285
  • 那天危彩,我揣著相機與錄音,去河邊找鬼泳桦。 笑死汤徽,一個胖子當著我的面吹牛,可吹牛的內容都是我干的灸撰。 我是一名探鬼主播谒府,決...
    沈念sama閱讀 38,389評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼浮毯!你這毒婦竟也來了完疫?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 37,019評論 0 259
  • 序言:老撾萬榮一對情侶失蹤债蓝,失蹤者是張志新(化名)和其女友劉穎壳鹤,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體饰迹,經...
    沈念sama閱讀 43,519評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡芳誓,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 35,971評論 2 325
  • 正文 我和宋清朗相戀三年余舶,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片锹淌。...
    茶點故事閱讀 38,100評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡匿值,死狀恐怖,靈堂內的尸體忽然破棺而出赂摆,到底是詐尸還是另有隱情挟憔,我是刑警寧澤,帶...
    沈念sama閱讀 33,738評論 4 324
  • 正文 年R本政府宣布烟号,位于F島的核電站曲楚,受9級特大地震影響,放射性物質發(fā)生泄漏褥符。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,293評論 3 307
  • 文/蒙蒙 一抚垃、第九天 我趴在偏房一處隱蔽的房頂上張望喷楣。 院中可真熱鬧,春花似錦鹤树、人聲如沸铣焊。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,289評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽曲伊。三九已至,卻和暖如春追他,著一層夾襖步出監(jiān)牢的瞬間坟募,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,517評論 1 262
  • 我被黑心中介騙來泰國打工邑狸, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留懈糯,地道東北人。 一個月前我還...
    沈念sama閱讀 45,547評論 2 354
  • 正文 我出身青樓单雾,卻偏偏與公主長得像赚哗,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子硅堆,可洞房花燭夜當晚...
    茶點故事閱讀 42,834評論 2 345