使用一种语言编写各种应用的时候,横亘在开发者面前的第一个问题就是如何进行状态管理。在前端领域,我们习惯使用框架或者各种辅助库来进行状态管理。例如,开发者经常使用react自带的context,或者mobx/redux等工具来管理组件间状态。在大热的跨端框架flutter中,笔者将对社区中使用广泛的provider框架进行介绍。
provider pub链接
官方文档宣称(本文基于4.0版本),provider是一个依赖注入和状态管理的混合工具,通过组件来构建组件。
provider有以下三个特点:
在pubspec.yaml文件中加入如下内容:
dependencies:
provider: ^4.0.0
然后执行命令flutter pub get
,安装到本地。
使用时只需在文件头部加上如下内容:
import 'package:provider/provider.dart';
如果我们想让某个变量能够被一个widget及其子widget所引用,我们需要将其暴露出来,典型写法如下:
Provider(
create: (_) => new MyModel(),
child: ...
)
如果要使用先前暴露的对象,可以这样操作
class Home extends StatelessWidget {
@override
Widget build(BuildContext context) {
MyModel yourValue = Provider.of(context)
return ...
}
}
Provider的构造方法可以嵌套使用
Provider(
create: (_) => Something(),
child: Provider(
create: (_) => SomethingElse(),
child: Provider(
create: (_) => AnotherThing(),
child: someWidget,
),
),
),
上述代码看起来过于繁琐,走入了嵌套地狱,好在provider给了更加优雅的实现
MultiProvider(
providers: [
Provider(create: (_) => Something()),
Provider(create: (_) => SomethingElse()),
Provider(create: (_) => AnotherThing()),
],
child: someWidget,
)
在3.0版本之后,有一种新的代理provider可供使用,ProxyProvider
能够将不同provider中的多个值整合成一个对象,并将其发送给外层provider,当所依赖的多个provider中的任意一个发生变化时,这个新的对象都会更新。下面的例子使用ProxyProvider
来构建了一个依赖其他provider提供的计数器的例子
Widget build(BuildContext context) {
return MultiProvider(
providers: [
ChangeNotifierProvider(create: (_) => Counter()),
ProxyProvider(
create: (_, counter, __) => Translations(counter.value),
),
],
child: Foo(),
);
}
class Translations {
const Translations(this._value);
final int _value;
String get title => 'You clicked $_value times';
}
可以通过各种不同的provider来应对具体的需求
Provider
最基础的provider,它会获取一个值并将它暴露出来ListenableProvider
用来暴露可监听的对象,该provider将会监听对象的改变以便及时更新组件状态ChangeNotifierProvider
ListerableProvider依托于ChangeNotifier的一个实现,它将会在需要的时候自动调用ChangeNotifier.dispose
方法ValueListenableProvider
监听一个可被监听的值,并且只暴露ValueListenable.value
方法StreamProvider
监听一个流,并且暴露出其最近发送的值FutureProvider
接受一个Future
作为参数,在这个Future
完成的时候更新依赖接下来笔者将以自己项目来举例provider的用法
首先定义一个基类,完成一些UI更新等通用工作
import 'package:provider/provider.dart';
class ProfileChangeNotifier extends ChangeNotifier {
Profile get _profile => Global.profile;
@override
void notifyListeners() {
Global.saveProfile(); //保存Profile变更
super.notifyListeners();
}
}
之后定义自己的数据类
class UserModle extends ProfileChangeNotifier {
String get user => _profile.user;
set user(String user) {
_profile.user = user;
notifyListeners();
}
bool get isLogin => _profile.isLogin;
set isLogin(bool value) {
_profile.isLogin = value;
notifyListeners();
}
String get avatar => _profile.avatar;
set avatar(String value) {
_profile.avatar = value;
notifyListeners();
}
这里通过set
和get
方法劫持对数据的获取和修改,在有相关改动发生时通知组件树同步状态。
在主文件中,使用provider
class MyApp extends StatelessWidget with CommonInterface {
MyApp({Key key, this.info}) : super(key: key);
final info;
// This widget is the root of your application.
@override
Widget build(BuildContext context) {
UserModle newUserModel = new UserModle();
return MultiProvider(
providers: [
// 用户信息
ListenableProvider.value(value: newUserModel),
],
child: ListenContainer(),
);
}
}
接下来,在所有的子组件中,如果需要使用用户的名字,只需Provider.of
即可,但是这样的写法看上去不够精简,每次调用时都需要写很长的一段开头Provider.of
很是繁琐,故而这里我们可以简单封装一个抽象类:
abstract class CommonInterface {
String cUser(BuildContext context) {
return Provider.of(context).user;
}
}
在子组件声明时,使用with
,来简化代码
class MyApp extends StatelessWidget with CommonInterface {
......
}
在使用时只需cUser(context)
即可。
class _FriendListState extends State with CommonInterface {
@override
Widget build(BuildContext context) {
return Text(cUser(context));
}
}
项目完整代码详见本人仓库
initState
中获取Provider会报错?initState() {
super.initState();
print(Provider.of(context).value);
}
要解决这个问题,要么使用其他生命周期方法(didChangeDependencies/build)
didChangeDependencies() {
super.didChangeDependencies();
final value = Provider.of(context).value;
if (value != this.value) {
this.value = value;
print(value);
}
}
或者指明你不在意这个值的更新,比如
initState() {
super.initState();
print(Provider.of(context, listen: false).value);
}
ChangeNotifier
的过程中,如果更新变量的值就会报出异常?ChangeNotifier
时,整个渲染树还处在创建过程中。initState() {
super.initState();
Provider.of(context).fetchSomething();
}
这是不允许的,因为组件的更新是即时生效的。
换句话来说如果某些组件在异步过程之前构建,某些组件在异步过程之后构建,这很有可能触发你应用中的UI表现不一致,这是不允许的。
为了解决这个问题,需要把你的异步过程放在能够等效的影响组件树的地方
class MyNotifier with ChangeNotifier {
MyNotifier() {
_fetchSomething();
}
Future _fetchSomething() async {}
}
initState() {
super.initState();
Future.microtask(() =>
Provider.of(context).fetchSomething(someValue);
);
}
ChangeNotifier
吗?Provider.value()
和StatefulWidget
结合起来使用,达到即刷新状态又同步UI的目的.class Example extends StatefulWidget {
const Example({Key key, this.child}) : super(key: key);
final Widget child;
@override
ExampleState createState() => ExampleState();
}
class ExampleState extends State {
int _count;
void increment() {
setState(() {
_count++;
});
}
@override
Widget build(BuildContext context) {
return Provider.value(
value: _count,
child: Provider.value(
value: this,
child: widget.child,
),
);
}
}
当需要读取状态时:
return Text(Provider.of(context).toString());
当需要改变状态时:
return FloatingActionButton(
onPressed: Provider.of(context).increment,
child: Icon(Icons.plus_one),
);
provider
暴露了许多细节api以便使用者封装自己的provider,它们包括:SingleChildCloneableWidget
、InheritedProvider
、DelegateWidget
、BuilderDelegate
、ValueDelegate
等Provider.of
来替代Consumer/Selector
.child
参数来保证组件树只会重建某个特定的部分Foo(
child: Consumer(
builder: (_, a, child) {
return Bar(a: a, child: child);
},
child: Baz(),
),
)
在以上例子中,当A
改变时,只有Bar
会重新渲染,Foo
和Baz
并不会进行不必要的重建。
为了更精细地控制,我们还可以使用Selector
来忽略某些不会影响组件数的改变。
Selector(
selector: (_, list) => list.length,
builder: (_, length, __) {
return Text('$length');
}
);
在这个例子中,组件只会在list的长度发生改变时才会重新渲染,其内部元素改变时并不会触发重绘。
8人点赞
日记本
作者:广兰路地铁
链接:https://www.jianshu.com/p/31499233c673
来源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。