从零构建可扩展 Flutter 应用:v1.0 → v2.0 全代码详解 -《已适配开源鸿蒙》

从零构建可扩展 Flutter 应用:v1.0 → v2.0 全代码详解 -《已适配开源鸿蒙》
在这里插入图片描述

v1.0 → v2.0 全代码详解

在这里插入图片描述

从零构建可扩展 Flutter 应用:v1.0 → v2.0 全代码详解

副标题:一个干净、模块化、支持 Riverpod 状态管理的生产级起点

在 Flutter 开发中,很多项目失败并非因为技术不行,而是初期架构缺失。本文将手把手带你从 最简可行应用(v1.0) 出发,逐步演进到 支持状态管理、主题切换、本地持久化的 v2.0 架构,并附上全部代码 + 逐行解释,助你打造一个真正面向未来的 Flutter 应用。


🧱 第一阶段:v1.0 —— 干净的基础骨架

✅ 目标

  • 最小可运行应用
  • 标准 Material Design 结构
  • 清晰目录划分
  • 无第三方依赖

📁 项目结构

lib/ ├── main.dart ├── app.dart ├── screens/home_screen.dart └── widgets/app_bar_title.dart 

1. lib/main.dart —— 应用入口

import'package:flutter/material.dart';import'app.dart';voidmain(){runApp(constMyApp());}
说明main() 是 Dart 程序入口。调用 runApp() 启动 Flutter 应用。不在此处写业务逻辑,仅负责启动根组件 MyApp,符合单一职责原则。

2. lib/app.dart —— App 根组件

import'package:flutter/material.dart';import'screens/home_screen.dart';classMyAppextendsStatelessWidget{constMyApp({super.key});static bool get isDarkMode =>false;// 预留:未来可动态控制@override Widget build(BuildContext context){returnMaterialApp( title:'My App', debugShowCheckedModeBanner:false, theme:ThemeData( useMaterial3:true, colorScheme: ColorScheme.fromSeed(seedColor: Colors.blue), brightness: Brightness.light,), darkTheme:ThemeData( useMaterial3:true, colorScheme: ColorScheme.fromSeed(seedColor: Colors.blue, brightness: Brightness.dark), brightness: Brightness.dark,), themeMode: isDarkMode ? ThemeMode.dark : ThemeMode.light, home:constHomeScreen(),);}}
说明:使用 MaterialApp 提供 Material Design 支持。定义 themedarkTheme,为深色模式做准备。home 指向主页面,路由系统预留位置(后续可替换为 routesGoRouter)。isDarkMode 是静态变量,便于未来替换为状态管理驱动的动态值

3. lib/widgets/app_bar_title.dart —— 可复用组件

import'package:flutter/material.dart';classAppBarTitleextendsStatelessWidget{final String title;constAppBarTitle({super.key, required this.title});@override Widget build(BuildContext context){returnText( title, style: Theme.of(context).textTheme.titleLarge?.copyWith( fontWeight: FontWeight.bold,),);}}
说明:将 AppBar 标题封装为独立 widget,提高复用性。使用 Theme.of(context) 获取当前主题样式,自动适配深浅色。通过 required this.title 强制传参,避免空值错误。

4. lib/screens/home_screen.dart —— 主页面

import'package:flutter/material.dart';import'../widgets/app_bar_title.dart';classHomeScreenextendsStatefulWidget{constHomeScreen({super.key});@override State<HomeScreen>createState()=>_HomeScreenState();}class_HomeScreenStateextendsState<HomeScreen>{ int _counter =0;void_incrementCounter(){setState((){ _counter++;});}@override Widget build(BuildContext context){returnScaffold( appBar:AppBar( title:constAppBarTitle(title:'My App'), centerTitle:true,), body:Center( child:Column( mainAxisAlignment: MainAxisAlignment.center, children:<Widget>[constText('Welcome to My App v1.0!', style:TextStyle(fontSize:20, fontWeight: FontWeight.w500),),constSizedBox(height:24),Text('You have pushed the button this many times:'),Text('$_counter', style: Theme.of(context).textTheme.headlineMedium,),constSizedBox(height:32), ElevatedButton.icon( onPressed: _incrementCounter, icon:constIcon(Icons.add), label:constText('Increment'),),],),), floatingActionButton:FloatingActionButton( onPressed: _incrementCounter, tooltip:'Increment', child:constIcon(Icons.add),),);}}
说明:使用 StatefulWidget 管理局部计数器状态。_counter 是私有变量,通过 setState 触发 UI 更新。页面结构清晰:AppBar + Body + FloatingActionButton。局限性:状态无法跨页面共享,重启后丢失——这正是 v2.0 要解决的问题。

⚙️ 第二阶段:v2.0 —— 引入 Riverpod,实现可扩展架构

✅ 升级目标

  • 状态全局可访问
  • 支持深色/浅色/系统主题动态切换
  • 计数器数据持久化(重启不丢失)
  • 解耦 UI 与逻辑

📦 新增依赖(pubspec.yaml

dependencies:flutter:sdk: flutter flutter_riverpod: ^2.5.1 # 状态管理shared_preferences: ^2.2.2 # 本地存储
运行命令:

这里也可以直接进行保存

在这里插入图片描述

📁 v2.0 项目结构

lib/ ├── main.dart ├── app.dart ├── constants/app_colors.dart (可选) ├── providers/ │ ├── app_theme_provider.dart │ └── counter_provider.dart ├── screens/home_screen.dart ├── widgets/ │ ├── app_bar_title.dart │ └── theme_toggle_button.dart └── utils/logger.dart (预留) 
在这里插入图片描述

1. lib/main.dart —— 包裹 ProviderScope

import'package:flutter/material.dart';import'package:flutter_riverpod/flutter_riverpod.dart';import'app.dart';voidmain()async{ WidgetsFlutterBinding.ensureInitialized();// 必须在使用 shared_preferences 前调用runApp(constProviderScope(// Riverpod 的根容器 child:MyApp(),),);}
说明ProviderScope 是 Riverpod 的上下文提供者,必须包裹整个应用WidgetsFlutterBinding.ensureInitialized() 是使用平台通道(如 SharedPreferences)的必要步骤。

2. lib/providers/app_theme_provider.dart —— 主题状态管理

import'package:flutter/material.dart';import'package:flutter_riverpod/flutter_riverpod.dart';enum AppThemeMode { light, dark, system }// 用户选择的主题模式(可持久化)final appThemeModeProvider = StateProvider<AppThemeMode>((ref){return AppThemeMode.light;// TODO: 未来从 SharedPreferences 读取});// 计算实际生效的 ThemeModefinal effectiveThemeModeProvider = Provider<ThemeMode>((ref){final mode = ref.watch(appThemeModeProvider);switch(mode){case AppThemeMode.light:return ThemeMode.light;case AppThemeMode.dark:return ThemeMode.dark;case AppThemeMode.system:return ThemeMode.system;}});
说明StateProvider 用于简单状态(如枚举、bool、int)。effectiveThemeModeProvider 是一个派生状态,根据用户选择计算出 ThemeMode未来可轻松扩展:将初始值从 SharedPreferences 读取,实现“记住用户偏好”。

3. lib/providers/counter_provider.dart —— 持久化计数器

import'package:flutter_riverpod/flutter_riverpod.dart';import'package:shared_preferences/shared_preferences.dart';final counterProvider = StateNotifierProvider<CounterNotifier, int>((ref){returnCounterNotifier();});classCounterNotifierextendsStateNotifier<int>{CounterNotifier():super(0){_loadFromPrefs();// 初始化时加载} Future<void>_loadFromPrefs()async{final prefs =await SharedPreferences.getInstance();final saved = prefs.getInt('counter')??0; state = saved;// 触发 UI 更新} Future<void>increment()async{ state++;// 自动通知监听者final prefs =await SharedPreferences.getInstance();await prefs.setInt('counter', state);// 保存} Future<void>reset()async{ state =0;final prefs =await SharedPreferences.getInstance();await prefs.setInt('counter', state);}}
说明StateNotifierProvider 适合管理复杂状态逻辑(如异步、验证、副作用)。state = newValue 会自动通知所有监听者重建 UI。数据在 increment()reset() 中自动持久化到设备存储。

4. lib/widgets/theme_toggle_button.dart —— 主题切换按钮

import'package:flutter/material.dart';import'package:flutter_riverpod/flutter_riverpod.dart';import'../providers/app_theme_provider.dart';classThemeToggleButtonextendsConsumerWidget{constThemeToggleButton({super.key});@override Widget build(BuildContext context, WidgetRef ref){final currentMode = ref.watch(appThemeModeProvider);return PopupMenuButton<AppThemeMode>( onSelected:(mode)=> ref.read(appThemeModeProvider.notifier).state = mode, itemBuilder:(context)=>[PopupMenuItem( value: AppThemeMode.light, child:Row(children:[Icon(Icons.light_mode, color: Colors.yellow[700]),constSizedBox(width:8),constText('Light'),]),),PopupMenuItem( value: AppThemeMode.dark, child:Row(children:[Icon(Icons.dark_mode, color: Colors.blue[300]),constSizedBox(width:8),constText('Dark'),]),),PopupMenuItem( value: AppThemeMode.system, child:Row(children:[Icon(Icons.settings, color: Colors.grey),constSizedBox(width:8),constText('System'),]),),], child:IconButton( icon: currentMode == AppThemeMode.dark ?constIcon(Icons.dark_mode):constIcon(Icons.light_mode), tooltip:'Switch theme',),);}}
说明ConsumerWidget 允许使用 ref.watch 监听状态变化。ref.read(...) 用于写入状态(不监听变化,性能更好)。图标根据当前主题动态切换,提升用户体验。

5. lib/app.dart —— 使用 Riverpod 主题

import'package:flutter/material.dart';import'package:flutter_riverpod/flutter_riverpod.dart';import'screens/home_screen.dart';import'providers/app_theme_provider.dart';classMyAppextendsConsumerWidget{constMyApp({super.key});@override Widget build(BuildContext context, WidgetRef ref){final themeMode = ref.watch(effectiveThemeModeProvider);// 监听主题变化returnMaterialApp( title:'My App', debugShowCheckedModeBanner:false, theme:ThemeData(useMaterial3:true, colorScheme: ColorScheme.fromSeed(seedColor: Colors.blue)), darkTheme:ThemeData( useMaterial3:true, colorScheme: ColorScheme.fromSeed(seedColor: Colors.blue, brightness: Brightness.dark),), themeMode: themeMode, home:constHomeScreen(),);}}
关键点MyAppStatelessWidget 变为 ConsumerWidget,以便监听 effectiveThemeModeProvider。当用户切换主题时,MaterialApp 自动重建,应用新主题。

6. lib/screens/home_screen.dart —— 使用 Riverpod 状态

import'package:flutter/material.dart';import'package:flutter_riverpod/flutter_riverpod.dart';import'../providers/counter_provider.dart';import'../widgets/app_bar_title.dart';import'../widgets/theme_toggle_button.dart';classHomeScreenextendsConsumerWidget{constHomeScreen({super.key});@override Widget build(BuildContext context, WidgetRef ref){final count = ref.watch(counterProvider);// 自动监听计数器变化returnScaffold( appBar:AppBar( title:constAppBarTitle(title:'My App v2.0'), centerTitle:true, actions:[constThemeToggleButton()],// 主题切换按钮), body:Center( child:Column( mainAxisAlignment: MainAxisAlignment.center, children:[constText('Welcome to My App!', style:TextStyle(fontSize:20, fontWeight: FontWeight.w500)),constSizedBox(height:24),constText('Counter (persisted):'),Text('$count', style: Theme.of(context).textTheme.headlineMedium),constSizedBox(height:32), ElevatedButton.icon( onPressed:()=> ref.read(counterProvider.notifier).increment(), icon:constIcon(Icons.add), label:constText('Increment'),), TextButton.icon( onPressed:()=> ref.read(counterProvider.notifier).reset(), icon:constIcon(Icons.refresh), label:constText('Reset'),),],),), floatingActionButton:FloatingActionButton( onPressed:()=> ref.read(counterProvider.notifier).increment(), tooltip:'Increment', child:constIcon(Icons.add),),);}}
关键改进:不再使用 StatefulWidget,UI 与状态完全解耦。ref.watch(counterProvider) 自动更新计数显示。按钮通过 ref.read(...).increment() 触发状态变更。重启应用后,计数器值依然保留!

测试

在这里插入图片描述


在这里插入图片描述

🔮 未来扩展路径

这个 v2.0 架构已为以下能力预留“插槽”:

功能扩展方式
多页面导航添加 go_router,在 app.dart 中配置
API 请求创建 api_provider.dart,使用 FutureProvider
用户登录新增 auth_provider.dart,管理 Token 和用户信息
国际化启用 flutter_localizations,添加 ARB 文件
日志监控utils/logger.dart 中封装 Sentry/Firebase

例如,添加天气 API 只需:

// providers/weather_provider.dartfinal weatherProvider = FutureProvider<Weather>((ref)async{final res =await http.get(Uri.parse('https://api.weather.com/...'));return Weather.fromJson(json.decode(res.body));});

然后在 UI 中优雅处理加载/错误/数据状态:

ref.watch(weatherProvider).when( data:(weather)=>Text('${weather.temp}°C'), loading:()=>CircularProgressIndicator(), error:(err, _)=>Text('Failed to load'),);

✅ 总结

  • v1.0 是一个规范、可交付的起点,避免“脏快糙”陷阱。
  • v2.0 通过 Riverpod + 模块化设计,实现了状态集中管理、UI 逻辑解耦、数据持久化。
  • 所有代码均可直接复制使用,并具备良好的扩展性。
好的架构不是一开始就复杂的,而是从第一天起就为“变化”做好准备。

📥 附录:完整文件清单

你可以按以下顺序创建文件:

  1. lib/main.dart
  2. lib/app.dart
  3. lib/constants/app_colors.dart(可选)
  4. lib/providers/app_theme_provider.dart
  5. lib/providers/counter_provider.dart
  6. lib/screens/home_screen.dart
  7. lib/widgets/app_bar_title.dart
  8. lib/widgets/theme_toggle_button.dart
  9. lib/utils/logger.dart(可留空)

然后更新 pubspec.yaml 并运行 flutter pub get


Read more

Flutter 三方库 system_settings 的鸿蒙化适配指南 - 实现应用内直达系统深度配置、支持通知权限、显示、声音与开发者选项一键跳转

Flutter 三方库 system_settings 的鸿蒙化适配指南 - 实现应用内直达系统深度配置、支持通知权限、显示、声音与开发者选项一键跳转

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 三方库 system_settings 的鸿蒙化适配指南 - 实现应用内直达系统深度配置、支持通知权限、显示、声音与开发者选项一键跳转 前言 在进行 Flutter for OpenHarmony 的精细化应用开发中,引导用户去开启必要的系统权限(如允许通知、修改屏幕自动旋转)是保障应用功能完整性的重要环。system_settings 是一个专注于将复杂的系统跳转 URI 语义化的库。它能让你的代码更具可读性,快速触达鸿蒙系统的各类深度设置页面。本文将指导大家如何在鸿蒙真机上实现高效率的跳转逻辑。 一、原理解析 / 概念介绍 1.1 基础原理 system_settings 的核心是利用鸿蒙系统的 Want 启动机制。它预置了大量标准化的设置页面标识符,通过简单的 MethodChannel 调用,请求鸿蒙的 AbilityContext

By Ne0inhk
【Linux】Linux基本使用和程序部署

【Linux】Linux基本使用和程序部署

🎬 那我掉的头发算什么:个人主页 🔥 个人专栏: 《javaSE》《数据结构》《数据库》《javaEE》 ⛺️待到苦尽甘来日 文章目录 * Linux环境搭建 * 环境搭建方式 * 使用云服务器 * 使用终端软件连接到Linux * Linux常用命令 * ls * pwd * cd * touch * cat * mkdir * rm * cp * mv * tail * vim * grep * ps * netstat * 搭建java部署环境 * apt * JDK * MYSQL * 部署web项目到Linux * 什么是部署 * 环境配置 * 构建项目并打包 * 上传jar包运行程序 * 杀死进程 Linux环境搭建 环境搭建方式 主要有四种: 1. 直接安装在物理机上。但是 Linux 桌面使用起来非常不友好。所以不建议。【不推荐】。 2. 使用虚拟机软件,

By Ne0inhk
Linux 动静态库完全指南:制作、使用、原理与实战

Linux 动静态库完全指南:制作、使用、原理与实战

🔥草莓熊Lotso:个人主页 ❄️个人专栏: 《C++知识分享》《Linux 入门到实践:零基础也能懂》 ✨生活是默默的坚持,毅力是永久的享受! 🎬 博主简介: 文章目录 * 前言: * 一. 库的基础认知:是什么?有哪些? * 1.1 库的本质 * 1.2 库的分类与系统位置 * 1.3 预备工作:自定义库源码 * 二. 静态库:编译时链接,独立运行 * 2.1 整体图示:理清思路 * 2.2 静态库制作流程(Makefile 自动化,更简便) * 2.3 静态库使用场景与命令 * 2.4 静态库核心特点 * 三. 动态库:运行时链接,

By Ne0inhk
Flutter 组件 http_requests 适配鸿蒙 HarmonyOS 实战:极简网络请求,构建边缘端轻量级 RESTful 通讯架构

Flutter 组件 http_requests 适配鸿蒙 HarmonyOS 实战:极简网络请求,构建边缘端轻量级 RESTful 通讯架构

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 组件 http_requests 适配鸿蒙 HarmonyOS 实战:极简网络请求,构建边缘端轻量级 RESTful 通讯架构 前言 在鸿蒙(OpenHarmony)生态迈向多端协同、涉及大量轻量级 IOT 设备(如智能穿戴、工业传感器)及微服务透传的背景下,如何实现快速、低开销的 HTTP 通讯,已成为决定应用“响应敏捷度”的关键工程要素。在鸿蒙设备这类强调内存精简与极速启动的环境下,如果应用依然无差别地引入像 Dio 这种包含复杂拦截器、适配器及多重缓存逻辑的“重型网络航母”,由于由于框架初始化开销大、内存足迹(Memory Footprint)偏深,极易由于由于“过度封装”导致边缘设备主线程的瞬间负载过高。 我们需要一种能够剥离样板代码、支持一键 JSON

By Ne0inhk