Flutter-状态管理-provider-ChangeNotifierProvider的用法

概述

ChangeNotifier 提供的ListenableProvider规范,会在需要时自动调用 ChangeNotifier.dispose

依赖

provider: ^6.0.2

实现流程

相关类介绍

  1. ChangeNotifierProvider :负责监听模型变化从而通知Consumer
  2. ChangeNotifier:模型类,状态发生改变时调用notifyListeners()
  3. Consumer:消费者类,收到通知负责重构UI

实践步骤

创建模型类

  • 继承或者混入ChangeNotifier
  • 更新时调用notifyListeners()

因为模型类使用了ChangeNotifier,当我们需要更改某个状态的时候就可以调用notifyListeners(),并且在调用它的任何时候,ChangeNotifierProvider都会收到通知并且Consumer消费者将重建UI。

class NotifierCounter with ChangeNotifier {
  int count = 0;

  void increment() {
    count++;
    _notification();
  }

  void decrement() {
    count--;
    _notification();
  }

  void _notification() {
    notifyListeners();
  }
}

需要被监听的页面顶层入口使用ChangeNotifierProvider widget

@override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: const Text('Provider Demo')),
      body: ChangeNotifierProvider(
        lazy: true,
        create: (BuildContext context) => NotifierCounter(),
        child: const ProvidersDemo(),
      ),
    );
  }

被监听的widget使用Consumer widget包裹

Consumer(
  builder: (BuildContext context, NotifierCounter counter, child) {
    return Text('${counter.count}',
        style: const TextStyle(fontSize: 45));
  },
),

更新状态

context.read<NotifierCounter>().increment();
或者:
Provider.of<NotifierCounter>(context, listen: false).increment()
-----------------------
read,listen: false 表示不会监听状态改变(Consumer会监听)

context.watch<T>(),widget 能够监听到 T 类型的 provider 发生的改变。(如果不
使用Consumer限定监听范围会导致刷新范围过大)

context.select<T,R>(R cb(T value)),允许 widget 只监听 T 上的一部分内容的改变。

需要注意的是context必须是ChangeNotifierProvidercontext,如果此时的context不是ChangeNotifierProvider的可以使用Builder widget包裹ChangeNotifierProvider下的根widget

@override
  Widget build(BuildContext context) {
    return ChangeNotifierProvider(
      create: (BuildContext context) => NotifierCounter(),
      child: Builder(
        builder: (context) =>Column(
        ...

计数器完整实例

在这里插入图片描述

void main() => runApp(const MaterialApp(home: SimpleApp()));

class SimpleApp extends StatelessWidget {
  const SimpleApp({Key? key}) : super(key: key);

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: const Text('Provider Demo')),
      body: ChangeNotifierProvider(
        lazy: true,
        create: (BuildContext context) => NotifierCounter(),
        child: const ProvidersDemo(),
      ),
    );
  }
}

class ProvidersDemo extends StatelessWidget {
  const ProvidersDemo({Key? key}) : super(key: key);

  @override
  Widget build(BuildContext context) {
    return Center(
      child: Column(mainAxisAlignment: MainAxisAlignment.center, children: [
        const Text('Counter',
            style: TextStyle(fontSize: 34, fontWeight: FontWeight.bold)),
        Consumer(
          builder: (BuildContext context, NotifierCounter counter, child) {
            return Text('${counter.count}',
                style: const TextStyle(fontSize: 45));
          },
        ),
        Row(
          mainAxisAlignment: MainAxisAlignment.center,
          children: [
            TextButton.icon(
                onPressed: () => context.read<NotifierCounter>().increment(),
                label: const Text('increment'),
                icon: const Icon(Icons.add)),
            TextButton.icon(
                onPressed: () => context.read<NotifierCounter>().decrement(),
                label: const Text('decrement'),
                icon: const Icon(Icons.remove))
          ],
        )
      ]),
    );
  }
}

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

吴唐人

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值