我正在使用BLoC从Firestore加载我的预设对象。这是我的Bloc模型:
class StatisticsBloc extends BlocBase {
List<Preset> _presets;
StreamController<List<Preset>> _presetsController = new StreamController();
Stream<List<Preset>> get getPresets => _presetsController.stream.asBroadcastStream();
StatisticsBloc() {
print('init Statistics Bloc');
_presets = [];
Firestore.instance.collection('Presets').snapshots().asBroadcastStream().listen(_onPresetsLoaded);
}
@override
void dispose() {
print('Disposed Statistics Bloc');
_presetsController.close();
}
void _onPresetsLoaded(QuerySnapshot data) {
_presets = [];
data.documents.forEach((DocumentSnapshot snap) {
Preset preset = Preset.fromDoc(snap);
_presets.add(preset);
});
_presetsController.sink.add(_presets);
}
}
然后我像这样显示List:
class StatisticsPage extends StatelessWidget {
StatisticsPage() {
print('Created StatisticsPage');
}
@override
Widget build(BuildContext context) {
final StatisticsBloc statisticsBloc = BlocProvider.of<StatisticsBloc>(context);
final List<Preset> _ = [];
print(statisticsBloc.getPresets.isBroadcast);
return Scaffold(
appBar: AppBar(
title: Text('Statistics'),
),
body: StreamBuilder(
stream: statisticsBloc.getPresets,
initialData: _,
builder: (BuildContext context, AsyncSnapshot<List<Preset>> snapshot) {
if (snapshot.hasData) {
return ListView(
children: snapshot.data.map((Preset preset) {
print(preset.name);
return new ListTile(
title: new Text(preset.name),
subtitle: new Text(preset.id),
);
}).toList(),
);
} else {
Text('No Data');
print('No Data');
}
}
)
);
}
}
问题是,我在Tabbar中显示了StatisticsPage
,因此当我切换标签并返回它时,它将构建多次。在第一次访问它可以工作,但当我切换标签并返回它,小部件得到重建,我得到错误:Bad state: Stream has already been listened to.
。我试图将getPresets
Stream声明为一个BroadcastStream,你可以在StatisitcsBloc
中看到,但这不起作用。
另外作为一个棘手的问题:有没有更好的方法来改变我从Firestore到Stream<QuerySnapshot>
的Stream<List<Presets>>
?
很容易,看看来自BehaviorSubject class的RxDart library。
默认情况下,BehaviorSubject是广播(又称热门)控制器,以便履行Rx主题合同。这意味着可以多次收听主题的流。
所以,只需换行
StreamController<List<Preset>> _presetsController = new StreamController();
至
StreamController<List<Preset>> _presetsController = new BehaviorSubject();
并删除所有
.asBroadcastStream()
而已!
在官方文档中它是not recommended to use asBroadcastStream()
创建流控制器的一种更危险的方法是通过asBroadcastStream()查看单订阅控制器。调用asBroadcastStream基本上告诉用户想要接管流的生命周期管理的单订阅流。与cancelOnError订阅者一起使用时,这很容易导致从未关闭的单流订阅,从而泄漏内存或资源。