flutter自定义InheritedProvider实现状态管理详解

 更新时间:2022年11月14日 16:53:19   作者:李小轰_Rex  
这篇文章主要为大家介绍了flutter自定义InheritedProvider实现状态管理详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

InheritedWidget简单数据驱动模型

基于InheritedWidget,实现简单的数据驱动模型,模型结构如下:

1. 数据存储

使用 InheritedWidget,新建 InheritedProvider

import 'package:flutter/material.dart';
class InheritedProvider<T> extends InheritedWidget {
  final T data;
  InheritedProvider({@required this.data, Widget child}) : super(child: child);
  @override
  bool updateShouldNotify(InheritedProvider<T> oldWidget) {
    //在此简单返回true,则每次更新都会调用依赖其的子孙节点的 didChangeDependencies
    return true;
  }
}

2. 变更通知

使用flutter自带的 ChangeNotifier 组件,封装工具类 ChangeNotifierProvider 用于响应数据变更,触发UI更新。

class ChangeNotifierProvider<T extends ChangeNotifier> extends StatefulWidget {
  final Widget child;
  final T data; //共享的数据模型,要求继承ChangeNotifier
  const ChangeNotifierProvider({
    Key key,
    this.child,
    this.data,
  }) : super(key: key);
  ///提供方法,子树获取共享数据
  static T of<T>(BuildContext context) {
    final provider =
        context.dependOnInheritedWidgetOfExactType<InheritedProvider<T>>();
    return provider.data;
  }
  @override
  _ChangeNotifierProviderState createState() =>
      _ChangeNotifierProviderState<T>();
}
class _ChangeNotifierProviderState<T extends ChangeNotifier>
    extends State<ChangeNotifierProvider<T>> {
  ///如果数据发生变化(model类调用 notifyListeners),重新构建InheritedProvider
  void update() {
    setState(() => {});
  }
  @override
  void didUpdateWidget(ChangeNotifierProvider<T> oldWidget) {
    ///当Provider更新时,如果新旧数据不"==",则解绑旧数据监听,同时添加新数据监听
    if (widget.data != oldWidget.data) {
      oldWidget.data.removeListener(update);
      widget.data.addListener(update);
    }
    super.didUpdateWidget(oldWidget);
  }
  @override
  void initState() {
    widget.data.addListener(update);
    super.initState();
  }
  @override
  void dispose() {
    widget.data.removeListener(update);
    super.dispose();
  }
  @override
  Widget build(BuildContext context) {
    return InheritedProvider<T>(
      data: widget.data,
      child: widget.child,
    );
  }
}

原理:使用InheritedWidget作为父节点,当数据源更新时,数据模型经过ChangeNotifier的关联,触发 _ChangeNotifierProviderState 的 setState方法,刷新UI。

封装 NotifyConsumer 直接显式调用 ChangeNotifierProvider.of 获取共享数据语义不明确,我们将其做一层简单的封装用于子节点获取数据:

///响应数据变化的消费者
class NotifyConsumer<T> extends StatelessWidget {
  final Widget Function(BuildContext context, T value) builder;
  const NotifyConsumer({Key key, @required this.builder}) : super(key: key);
  @override
  Widget build(BuildContext context) {
    return builder(
      context,
      ChangeNotifierProvider.of<T>(context),
    );
  }
}

3. 使用方法

创建一个共享数据模型:

///共享数据模型
class CountModel extends ChangeNotifier {
  int _count = 0;
  int get count => _count;
  //提供自增方法
  void increase() {
    _count++;
    // 通知监听器(订阅者),重新构建InheritedProvider, 更新状态。
    notifyListeners();
  }
}

ChangeNotifierProvider 的使用方式:

import 'package:flutter/material.dart';
import 'package:test_interited/provider/change_notifier_provider.dart';
import 'package:test_interited/provider/notify_consumer.dart';
import 'package:test_interited/provider/test/count_model.dart';
class TestNotifierWidget extends StatefulWidget {
  const TestNotifierWidget({Key key}) : super(key: key);
  @override
  _TestNotifierWidgetState createState() => _TestNotifierWidgetState();
}
class _TestNotifierWidgetState extends State<TestNotifierWidget> {
  @override
  Widget build(BuildContext context) {
    return Center(
        child: ChangeNotifierProvider<CountModel>(
            data: CountModel(),
            child: Builder(builder: (context) {
              return Column(
                mainAxisAlignment: MainAxisAlignment.center,
                children: [
                  Builder(
                    builder: (context) {
                      //获取共享数据源
                      return NotifyConsumer<CountModel>(
                          builder: (context, value) {
                        return Text(value.count.toString());
                      });
                    },
                  ),
                  Builder(builder: (context) {
                    return RaisedButton(
                      child: Text('自增'),
                      onPressed: () {
                        //获取共享model,进行数据自增操作
                        ChangeNotifierProvider.of<CountModel>(context).increase();
                      },
                    );
                  }),
                ],
              );
            })));
  }
}

以上就是flutter自定义InheritedProvider实现状态管理详解的详细内容,更多关于flutter自定义InheritedProvider的资料请关注脚本之家其它相关文章!

相关文章

  • Android新建水平节点进度条示例

    Android新建水平节点进度条示例

    这篇文章主要为大家介绍了Android新建水平节点进度条示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-06-06
  • Android ListView长按弹出菜单二种实现方式示例

    Android ListView长按弹出菜单二种实现方式示例

    这篇文章主要介绍了Android ListView长按弹出菜单的方法,大家参考实现
    2013-11-11
  • Kotlin协程概念原理与使用万字梳理

    Kotlin协程概念原理与使用万字梳理

    协程的作用是什么?协程是一种轻量级的线程,解决异步编程的复杂性,异步的代码使用协程可以用顺序进行表达,文中通过示例代码介绍详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习吧
    2022-08-08
  • Android自定义渐变式炫酷ListView下拉刷新动画

    Android自定义渐变式炫酷ListView下拉刷新动画

    这篇文章主要为大家详细介绍了Android自定义渐变式炫酷ListView下拉刷新动画,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-02-02
  • Android实现动态高斯模糊效果

    Android实现动态高斯模糊效果

    在Android开发中常常会用到高斯模糊,但有的时候我们可能会需要一个图片以不同的模糊程度展现出来,那如何实现呢,一起通过本文来学习学习吧。
    2016-08-08
  • Android实现手机震动抖动效果的方法

    Android实现手机震动抖动效果的方法

    今天小编就为大家分享一篇关于Android实现手机震动抖动效果的方法,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2019-03-03
  • Android仿百度地图小度语音助手的贝塞尔曲线动画

    Android仿百度地图小度语音助手的贝塞尔曲线动画

    这篇文章主要为大家详细介绍了Android仿百度地图小度语音助手的贝塞尔曲线动画,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-07-07
  • Android中常见的图形绘制方式总结

    Android中常见的图形绘制方式总结

    Android中绘制图片或形状是我们常遇到的事情,通过最近的学习与在网上学习的案例与资料那么我今天就总结一下,这篇文章主要给大家介绍了关于Android中常见的图形绘制方式,需要的朋友可以参考下
    2021-07-07
  • Android 实现单线程轮循机制批量下载图片

    Android 实现单线程轮循机制批量下载图片

    这篇文章主要介绍了Android 单线程轮循机制批量下载图片的相关资料,这里对实现步骤做了详细介绍,需要的朋友可以参考下
    2017-07-07
  • ReactNative Alert详解及实例代码

    ReactNative Alert详解及实例代码

    这篇文章主要介绍了ReactNative Alert详解及实例代码的相关资料,需要的朋友可以参考下
    2016-10-10

最新评论