当前位置: 首页 > news >正文

Flutter 中的数据跨层传递方案

在 Flutter 中,数据跨层传递(从父组件向子组件传递数据,或从子组件向父组件传递)有多种方案,主要包括以下几种:


1. 直接参数传递(Constructor 参数)

适用场景

  • 父组件向子组件传递数据,适用于层级关系较浅的情况。

示例

class ParentWidget extends StatelessWidget {final String message = "Hello from Parent";Widget build(BuildContext context) {return ChildWidget(message: message);}
}class ChildWidget extends StatelessWidget {final String message;const ChildWidget({required this.message});Widget build(BuildContext context) {return Text(message);}
}

优点:简单、直观。
缺点:如果层级较深,可能需要逐层传递,导致“参数传递链”问题。


2. InheritedWidget(Flutter 低级状态管理方案)

适用场景

  • 子 Widget 跨层共享父 Widget 的属性,且数据不会频繁更新。

示例

class MyDataProvider extends InheritedWidget {final String data;const MyDataProvider({required this.data, required Widget child}) : super(child: child);static MyDataProvider? of(BuildContext context) {return context.dependOnInheritedWidgetOfExactType<MyDataProvider>();}bool updateShouldNotify(covariant MyDataProvider oldWidget) {return oldWidget.data != data;}
}class ParentWidget extends StatelessWidget {Widget build(BuildContext context) {return MyDataProvider(data: "Hello from InheritedWidget",child: ChildWidget(),);}
}class ChildWidget extends StatelessWidget {Widget build(BuildContext context) {final provider = MyDataProvider.of(context);return Text(provider?.data ?? "No data");}
}

优点:数据可在组件树中被多个组件访问,无需手动传递。
缺点:手写较繁琐,通常与 ChangeNotifierProvider 结合使用。


3. Provider(推荐)

适用场景

  • 全局状态管理,适用于多个页面共享数据的情况。

示例

import 'package:flutter/material.dart';
import 'package:provider/provider.dart';class CounterProvider with ChangeNotifier {int count = 0;void increment() {count++;notifyListeners();}
}void main() {runApp(ChangeNotifierProvider(create: (_) => CounterProvider(),child: MyApp(),),);
}class MyApp extends StatelessWidget {Widget build(BuildContext context) {return MaterialApp(home: CounterPage(),);}
}class CounterPage extends StatelessWidget {Widget build(BuildContext context) {return Scaffold(appBar: AppBar(title: Text("Provider Example")),body: Column(mainAxisAlignment: MainAxisAlignment.center,children: [Text("Count: ${context.watch<CounterProvider>().count}"),ElevatedButton(onPressed: () {context.read<CounterProvider>().increment();},child: Text("Increment"),),],),);}
}

优点:官方推荐、简洁、支持自动通知 UI 更新。
缺点:需要引入 provider 包。


4. Riverpod(增强版 Provider)

适用场景

  • 需要更强的依赖注入和状态管理能力,避免 Provider 的嵌套问题。

示例

import 'package:flutter/material.dart';
import 'package:flutter_riverpod/flutter_riverpod.dart';final counterProvider = StateProvider<int>((ref) => 0);void main() {runApp(ProviderScope(child: MyApp()));
}class MyApp extends StatelessWidget {Widget build(BuildContext context) {return MaterialApp(home: CounterPage(),);}
}class CounterPage extends ConsumerWidget {Widget build(BuildContext context, WidgetRef ref) {final count = ref.watch(counterProvider);return Scaffold(appBar: AppBar(title: Text("Riverpod Example")),body: Column(mainAxisAlignment: MainAxisAlignment.center,children: [Text("Count: $count"),ElevatedButton(onPressed: () => ref.read(counterProvider.notifier).state++,child: Text("Increment"),),],),);}
}

优点:比 Provider 更简洁、更强大,避免 context 依赖问题。
缺点:学习成本较高,需要使用 flutter_riverpod 包。


5. GetX(轻量级但强大的状态管理)

适用场景

  • 想要减少模板代码,并需要更高性能的状态管理。

示例

import 'package:flutter/material.dart';
import 'package:get/get.dart';class CounterController extends GetxController {var count = 0.obs;void increment() => count++;
}void main() {runApp(GetMaterialApp(home: CounterPage()));
}class CounterPage extends StatelessWidget {final CounterController controller = Get.put(CounterController());Widget build(BuildContext context) {return Scaffold(appBar: AppBar(title: Text("GetX Example")),body: Column(mainAxisAlignment: MainAxisAlignment.center,children: [Obx(() => Text("Count: ${controller.count}")),ElevatedButton(onPressed: controller.increment,child: Text("Increment"),),],),);}
}

优点:无需 context,代码简洁,性能高。
缺点:非官方方案,项目可能不易维护。


6. EventBus(跨组件间事件通信)

适用场景

  • 非父子组件之间传递数据,比如 多个页面或多个独立的组件 之间通信。

示例

import 'package:event_bus/event_bus.dart';final EventBus eventBus = EventBus();class DataEvent {final String data;DataEvent(this.data);
}// 发送事件
eventBus.fire(DataEvent("Hello EventBus"));// 监听事件
eventBus.on<DataEvent>().listen((event) {print(event.data);
});

优点:适合全局事件通信,避免组件嵌套传参。
缺点:管理不当可能导致事件滥用,影响代码可读性。


7. Notification(从子组件向父组件传递数据)

适用场景

  • 适用于 子组件向父组件发送事件通知,常用于 滚动监听交互事件 等情况。

示例

子组件 发送 CustomNotification,父组件 监听 并接收通知:

import 'package:flutter/material.dart';// 自定义通知类
class CustomNotification extends Notification {final String message;CustomNotification(this.message);
}class NotificationExample extends StatelessWidget {Widget build(BuildContext context) {return Scaffold(appBar: AppBar(title: Text("Notification Example")),body: NotificationListener<CustomNotification>(onNotification: (notification) {ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text("收到通知: ${notification.message}")),);return true; // 返回 true,表示已处理通知},child: ChildWidget(),),);}
}class ChildWidget extends StatelessWidget {Widget build(BuildContext context) {return Center(child: ElevatedButton(onPressed: () {CustomNotification("Hello, Parent!").dispatch(context);},child: Text("发送通知"),),);}
}void main() {runApp(MaterialApp(home: NotificationExample()));
}

解析

  • NotificationListener<CustomNotification> 监听 子组件 ChildWidget 发送的 CustomNotification 事件。
  • CustomNotification("Hello, Parent!").dispatch(context); 从子组件向上传递通知,并触发 onNotification 方法。
  • ScaffoldMessenger.of(context).showSnackBar(...) 在父组件显示通知

优点

  • 适合 从子组件向上 传递事件,而无需父组件主动监听或传递回调函数。
  • 无需 Provider、全局状态,适用于简单的事件通知。

缺点

  • 不适合传递复杂数据(比如全局状态)。
  • 仅能向上级组件传递,不能横向或全局广播数据。

使用场景

  • 子组件通知父组件某些事件发生(如按钮点击、用户输入变化)。
  • 监听滚动事件ScrollNotification)。
  • 监听输入框变化FormField 自带 FormFieldState)。

总结

方案方向适用场景优点缺点
构造函数传参父 → 子普通数据传递简单直观层级深时麻烦
InheritedWidget父 → 子共享数据但不频繁变更内置方案代码较复杂
Provider全局官方推荐,适合全局状态易用高效需要第三方库
Riverpod全局更强大的依赖管理简单、无 context 限制学习成本
GetX全局轻量级状态管理代码简洁,性能好非官方方案
EventBus任意跨组件通信适合全局事件可能滥用
Notification子 → 父子组件通知父组件无需回调,适合单向事件仅支持向上传递

结论

  • 状态管理:推荐 Provider / Riverpod
  • 子 → 父传递Notification(事件通知),或者直接 回调Function)。
  • 轻量级方案GetX(简洁),或者 直接 StatefulWidget 维护本地状态。

相关文章:

Flutter 中的数据跨层传递方案

在 Flutter 中&#xff0c;数据跨层传递&#xff08;从父组件向子组件传递数据&#xff0c;或从子组件向父组件传递&#xff09;有多种方案&#xff0c;主要包括以下几种&#xff1a; 1. 直接参数传递&#xff08;Constructor 参数&#xff09; 适用场景&#xff1a; 父组件向…...

代码随想录D50-51 图论 Python

理论基础 理论基础部分依然沿用代码随想录教程中的介绍&#xff1a; 图的种类 度 连通性 连通性用于表示图中节点的连通情况。 如果有节点不能到达其他节点&#xff0c;则为非连通图&#xff0c;想象将多个水分子表示为图&#xff0c;不考虑非键作用&#xff0c;这张图就不是…...

MyBatis进阶

日志的使用 我们在使用MyBatis的时候, 其实MyBatis框架会打印一些必要的日志信息, 在开发阶段这些日志信息对我们分析问题,理解代码的执行是特别有帮助的; 包括项目上线之后,我们也可以收集项目的错误日志到文件里面去; 所以我们采用专门的日志系统来处理. 步骤 导入坐标拷贝…...

容器化部署Kafka的最佳实践:基于KRaft模式的无ZooKeeper方案

一、docker 部署kafka单节点 1.1安装docker 可以参考这篇CentOS 7安装docker并配置镜像加速 1.3 运行kafka&#xff08;注意修改zookeeper&#xff0c;kafka地址&#xff09; docker run -d --name kafka -e KAFKA_ADVERTISED_LISTENERSPLAINTEXT://172.16.10.180:9092 -p …...

DeepSeek核心算法解析:如何打造比肩ChatGPT的国产大模型

注&#xff1a;此文章内容均节选自充电了么创始人&#xff0c;CEO兼CTO陈敬雷老师的新书《自然语言处理原理与实战》&#xff08;人工智能科学与技术丛书&#xff09;【陈敬雷编著】【清华大学出版社】 文章目录 DeepSeek大模型技术系列一DeepSeek核心算法解析&#xff1a;如何…...

LeetCode-633. 平方数之和

1、题目描述 给定一个非负整数 c &#xff0c;你要判断是否存在两个整数 a 和 b&#xff0c;使得 a2 b2 c 。 示例 1&#xff1a; 输入&#xff1a;c 5 输出&#xff1a;true 解释&#xff1a;1 * 1 2 * 2 5示例 2&#xff1a; 输入&#xff1a;c 3 输出&#xff1a;f…...

前端面试技巧与实践

在当今快速发展的互联网行业中&#xff0c;前端开发已经成为了一个至关重要的角色。随着技术的不断进步和用户需求的日益复杂&#xff0c;前端工程师的职责不再仅仅是实现页面的布局和交互&#xff0c;而是需要具备全方位的技术能力和工程思维。根据2023年Stack Overflow的开发…...

windows Redis Insight 如何查看宝塔docker里的redis数据

1、ping 命令用于测试网络连通性&#xff0c;它只需要目标 IP 地址作为参数&#xff0c;不需要端口号。正确的命令如下&#xff1a; ping 公网地址2、使用 Telnet 测试端口连通性 telnet 公网地址 端口 telnet 47.108.67.228 6379如果连接成功&#xff0c;窗口会变为空白&am…...

sql数据执行失败,三个命令依次执行

set global innodb_strict_mode off set global.sql_mode ONLY_FULL_GROUP_BY,STRICT_TRANS_TABLES,ERROR_FOR_DIVISION_BY_ZERO,NO_ENGINE_SUBSTITUTION; set sql_mode ONLY_FULL_GROUP_BY,STRICT_TRANS_TABLES,ERROR_FOR_DIVISION_BY_ZERO,NO_ENGINE_SUBSTITUTION;...

BGP配置华为——RR反射器配置

实验拓扑 与之前实验同理将loop0作为routerID使用&#xff0c;且R1和R2上用loop1接口用于模拟用户其他网段 实验要求 1&#xff0c;在AS100内运行OSPF协议 2.配置路由反射器&#xff0c;使得从R1进入的数据能够反射到全局网络 3.在R1和R2上分别宣告自己的loop1口网段用于观…...

基于Flask的艺恩影片票房分析系统的设计与实现

【Flask】基于Flask的艺恩影片票房分析系统的设计与实现&#xff08;完整系统源码开发笔记详细部署教程&#xff09;✅ 目录 一、项目简介二、项目界面展示三、项目视频展示 一、项目简介 该系统利用Python编程语言进行后端开发&#xff0c;结合Echarts进行数据的可视化展示&a…...

架构设计系列(三):架构模式

一、概述 关于移动应用开发中常见的架构模式&#xff0c;这些模式是为了克服早期模式的局限性而引入。常见的 架构模式有&#xff1a; MVC, MVP, MVVM, MVVM-C, and VIPER 二、MVC, MVP, MVVM, MVVM-C, and VIPER架构模式 MVC、MVP、MVVM、MVVM-C 和 VIPER 是移动应用开发中…...

零基础学QT、C++(一)安装QT

目录 如何快速学习QT、C呢&#xff1f; 一、编译器、项目构建工具 1、编译器&#xff08;介绍2款&#xff09; 2、项目构建工具 二、安装QT 1、下载QT安装包 2、运行安装包 3、运行QT creator 4、导入开源项目 总结 闲谈 如何快速学习QT、C呢&#xff1f; 那就是项目驱动法&…...

SQL注入(SQL Injection)详解与实战

文章目录 一、什么是SQL注入&#xff1f;二、常见SQL注入类型三、手动注入步骤&#xff08;以CTF题目为例&#xff09;四、CTF实战技巧五、自动化工具&#xff1a;SQLMap六、防御措施七、CTF例题八、资源推荐 一、什么是SQL注入&#xff1f; SQL注入是一种通过用户输入构造恶意…...

【Prometheus】prometheus结合domain_exporter实现域名监控

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全…...

Java 设计模式之命令模式

文章目录 Java 设计模式之命令模式概述UML代码实现简单实现复杂实现 Java 设计模式之命令模式 概述 命令模式(Command)&#xff1a;将一个请求封装为一个对象&#xff0c;从而使你可用不同的请求对客户进行参数化&#xff1b;对请求排队或记录请求日志&#xff0c;以及支持可…...

BT401双模音频蓝牙模块如何开启ble的透传,有什么注意事项

BT401音频蓝牙模块如何开启ble的透传&#xff1f; 首先BT401的蓝牙音频模块&#xff0c;分为两个版本&#xff0c;dac版本和iis数字音频版本 DAC版本&#xff1a;就是BT401蓝牙模块【9和10脚】直接输出模拟音频信号&#xff0c;也就是说&#xff0c;直接推动耳机可以听到声音 …...

利用二分法+布尔盲注、时间盲注进行sql注入

一、布尔盲注&#xff1a; import requestsdef binary_search_character(url, query, index, low32, high127):while low < high:mid (low high 1) // 2payload f"1 AND ASCII(SUBSTRING(({query}),{index},1)) > {mid} -- "res {"id": payloa…...

Vue 项目登录的基本流程

Vue 用户登录的基本流程包括以下6个步骤&#xff1a; 步骤&#xff1a; 1. 创建登录表单 在前端&#xff0c;首先要创建一个登录表单&#xff0c;用户输入账号&#xff08;用户名、邮箱、手机号等&#xff09;和密码。 示例&#xff1a;Login.vue <template><div…...

kubernetes源码分析 kubelet

简介 从官方的架构图中很容易就能找到 kubelet 执行 kubelet -h 看到 kubelet 的功能介绍&#xff1a; kubelet 是每个 Node 节点上都运行的主要“节点代理”。使用如下的一个向 apiserver 注册 Node 节点&#xff1a;主机的 hostname&#xff1b;覆盖 host 的参数&#xff1…...

PHP和Node.js哪个更爽?

先说结论&#xff0c;rust完胜。 php&#xff1a;laravel&#xff0c;swoole&#xff0c;webman&#xff0c;最开始在苏宁的时候写了几年php&#xff0c;当时觉得php真的是世界上最好的语言&#xff0c;因为当初活在舒适圈里&#xff0c;不愿意跳出来&#xff0c;就好比当初活在…...

前端倒计时误差!

提示:记录工作中遇到的需求及解决办法 文章目录 前言一、误差从何而来?二、五大解决方案1. 动态校准法(基础版)2. Web Worker 计时3. 服务器时间同步4. Performance API 高精度计时5. 页面可见性API优化三、生产环境最佳实践四、终极解决方案架构前言 前几天听说公司某个项…...

关于nvm与node.js

1 安装nvm 安装过程中手动修改 nvm的安装路径&#xff0c; 以及修改 通过nvm安装node后正在使用的node的存放目录【这句话可能难以理解&#xff0c;但接着往下看你就了然了】 2 修改nvm中settings.txt文件配置 nvm安装成功后&#xff0c;通常在该文件中会出现以下配置&…...

SpringBoot+uniapp 的 Champion 俱乐部微信小程序设计与实现,论文初版实现

摘要 本论文旨在设计并实现基于 SpringBoot 和 uniapp 的 Champion 俱乐部微信小程序&#xff0c;以满足俱乐部线上活动推广、会员管理、社交互动等需求。通过 SpringBoot 搭建后端服务&#xff0c;提供稳定高效的数据处理与业务逻辑支持&#xff1b;利用 uniapp 实现跨平台前…...

深入解析C++中的extern关键字:跨文件共享变量与函数的终极指南

&#x1f680; C extern 关键字深度解析&#xff1a;跨文件编程的终极指南 &#x1f4c5; 更新时间&#xff1a;2025年6月5日 &#x1f3f7;️ 标签&#xff1a;C | extern关键字 | 多文件编程 | 链接与声明 | 现代C 文章目录 前言&#x1f525;一、extern 是什么&#xff1f;&…...

第 86 场周赛:矩阵中的幻方、钥匙和房间、将数组拆分成斐波那契序列、猜猜这个单词

Q1、[中等] 矩阵中的幻方 1、题目描述 3 x 3 的幻方是一个填充有 从 1 到 9 的不同数字的 3 x 3 矩阵&#xff0c;其中每行&#xff0c;每列以及两条对角线上的各数之和都相等。 给定一个由整数组成的row x col 的 grid&#xff0c;其中有多少个 3 3 的 “幻方” 子矩阵&am…...

【Go语言基础【13】】函数、闭包、方法

文章目录 零、概述一、函数基础1、函数基础概念2、参数传递机制3、返回值特性3.1. 多返回值3.2. 命名返回值3.3. 错误处理 二、函数类型与高阶函数1. 函数类型定义2. 高阶函数&#xff08;函数作为参数、返回值&#xff09; 三、匿名函数与闭包1. 匿名函数&#xff08;Lambda函…...

人机融合智能 | “人智交互”跨学科新领域

本文系统地提出基于“以人为中心AI(HCAI)”理念的人-人工智能交互(人智交互)这一跨学科新领域及框架,定义人智交互领域的理念、基本理论和关键问题、方法、开发流程和参与团队等,阐述提出人智交互新领域的意义。然后,提出人智交互研究的三种新范式取向以及它们的意义。最后,总结…...

libfmt: 现代C++的格式化工具库介绍与酷炫功能

libfmt: 现代C的格式化工具库介绍与酷炫功能 libfmt 是一个开源的C格式化库&#xff0c;提供了高效、安全的文本格式化功能&#xff0c;是C20中引入的std::format的基础实现。它比传统的printf和iostream更安全、更灵活、性能更好。 基本介绍 主要特点 类型安全&#xff1a…...

DeepSeek源码深度解析 × 华为仓颉语言编程精粹——从MoE架构到全场景开发生态

前言 在人工智能技术飞速发展的今天&#xff0c;深度学习与大模型技术已成为推动行业变革的核心驱动力&#xff0c;而高效、灵活的开发工具与编程语言则为技术创新提供了重要支撑。本书以两大前沿技术领域为核心&#xff0c;系统性地呈现了两部深度技术著作的精华&#xff1a;…...