flutter状态管理之BloC


一、概述

Bloc = Business Logic Component

二、结构

BloC本身是一个独立的package、flutter-bloc基于BloC和Provider实现,其结构如下图所示:

三、 实现

3.1 BloC是基于dart的Stream API实现的发布订阅模式。
  • 通过on方法订阅事件并提供处理函数,收到Event再将Event转换成State返回给Bloc
  • 通过add方法发布事件
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
abstract class Bloc<Event, State> extends BlocBase<State>
implements BlocEventSink<Event> {
//所有的订阅者
final _subscriptions = <StreamSubscription<dynamic>>[];

final _eventController = StreamController<Event>.broadcast();

//添加订阅者
void on<E extends Event>(
EventHandler<E, State> handler, {
EventTransformer<E>? transformer,
}) {
//这里开始了listen
final subscription = xxx.listen(null);
_subscriptions.add(subscription);
}

@override
void add(Event event) {
try {
onEvent(event);
_eventController.add(event);
} catch (error, stackTrace) {
onError(error, stackTrace);
rethrow;
}
}

}
3.1 事件和状态之间的转换
  • 转换逻辑有业务层通过on()注入
  • 内部的核心是_Emitter以及 _eventController_stateController
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
(dynamic event) {
void onEmit(State state) {
if (isClosed) return;
if (this.state == state && _emitted) return;
onTransition(Transition(
currentState: this.state,
event: event as E,
nextState: state,
));
emit(state);
}

final emitter = _Emitter(onEmit);
final controller = StreamController<E>.broadcast(
sync: true,
onCancel: emitter.cancel,
);

void handleEvent() async {
void onDone() {
emitter.complete();
_emitters.remove(emitter);
if (!controller.isClosed) controller.close();
}

try {
_emitters.add(emitter);
//交给外部实现转换事件和状态
await handler(event as E, emitter);
} catch (error, stackTrace) {
onError(error, stackTrace);
rethrow;
} finally {
onDone();
}
}

handleEvent();
return controller.stream;
}

四、参考

主要参考和debug了bloc仓库里面examples中login的例子:
这里