flutter-riverpod-expert by juparave/dotfiles
npx skills add https://github.com/juparave/dotfiles --skill flutter-riverpod-expert您拥有遵循 2025 年最佳实践的 Flutter Riverpod 状态管理专业知识。当用户处理 Riverpod 或 Flutter 状态管理时,请应用这些模式和指南。
当用户提及以下内容时,激活此专业知识:
@riverpod 注解和 riverpod_generatorselect() 来优化重建不可变/计算值 - 使用 Provider:
You have expert knowledge in Flutter Riverpod state management following 2025 best practices. When the user is working with Riverpod or Flutter state management, apply these patterns and guidelines.
Activate this expertise when the user mentions:
@riverpod annotations and riverpod_generatorselect() to optimize rebuilds广告位招租
在这里展示您的产品或服务
触达数万 AI 开发者,精准高效
@riverpod
String apiKey(Ref ref) => 'YOUR_API_KEY';
@riverpod
int totalPrice(Ref ref) {
final cart = ref.watch(cartProvider);
return cart.items.fold(0, (sum, item) => sum + item.price);
}
简单的同步状态 - 使用 NotifierProvider:
@riverpod
class Counter extends _$Counter {
@override
int build() => 0;
void increment() => state++;
void decrement() => state = max(0, state - 1);
}
带有数据变更的异步数据 (2025 年首选) - 使用 AsyncNotifierProvider:
@riverpod
class TodoList extends _$TodoList {
@override
Future<List<Todo>> build() async {
final repo = ref.watch(todoRepositoryProvider);
return repo.fetchTodos();
}
Future<void> addTodo(String title) async {
state = const AsyncLoading();
state = await AsyncValue.guard(() async {
final repo = ref.read(todoRepositoryProvider);
await repo.createTodo(title);
return repo.fetchTodos();
});
}
Future<void> deleteTodo(String id) async {
// 乐观更新
state = AsyncData(state.value!.where((t) => t.id != id).toList());
try {
await ref.read(todoRepositoryProvider).deleteTodo(id);
} catch (e) {
ref.invalidateSelf(); // 出错时回滚
}
}
}
仅用于实时流 - 使用 StreamProvider:
@riverpod
Stream<User?> authState(Ref ref) {
return FirebaseAuth.instance.authStateChanges();
}
关键规则:为了更好的一致性和数据变更支持,优先使用 AsyncNotifierProvider 而不是 FutureProvider/StreamProvider。
dependencies:
flutter_riverpod: ^2.5.0
riverpod_annotation: ^2.3.0
dev_dependencies:
build_runner: ^2.4.0
riverpod_generator: ^2.4.0
custom_lint: ^0.6.0
riverpod_lint: ^2.3.0
每个提供者文件都需要:
import 'package:riverpod_annotation/riverpod_annotation.dart';
part 'filename.g.dart'; // 必需
@riverpod
class MyProvider extends _$MyProvider {
@override
Future<Data> build() async => fetchData();
}
# 监视模式 (开发期间推荐)
dart run build_runner watch -d
# 一次性生成
dart run build_runner build --delete-conflicting-outputs
// ❌ 不好:产品有任何变化都会重建
final product = ref.watch(productProvider);
return Text('\$${product.price}');
// ✅ 好:仅当价格变化时重建
final price = ref.watch(productProvider.select((p) => p.price));
return Text('\$$price');
ref.watch() - 订阅变化(在 build 中使用):
@Override
Widget build(BuildContext context, WidgetRef ref) {
final todos = ref.watch(todoListProvider);
return ListView(...);
}
ref.select() - 订阅特定属性(优化重建):
final count = ref.watch(todoListProvider.select((todos) => todos.length));
final isAdult = ref.watch(personProvider.select((p) => p.age >= 18));
ref.read() - 一次性读取,不订阅(仅用于事件处理程序):
onPressed: () {
ref.read(todoListProvider.notifier).addTodo('New task');
}
// ❌ 切勿在 build 中使用 read() 来"优化" - 它不会触发重建!
ref.listen() - 副作用(导航、SnackBar、日志记录):
ref.listen<AsyncValue<List<Todo>>>(
todoListProvider,
(previous, next) {
next.whenOrNull(
error: (error, stack) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(content: Text('Error: $error')),
);
},
);
},
);
// ❌ 不好:导致性能问题
ListView.builder(
itemBuilder: (context, index) {
final todo = ref.watch(todoProvider(ids[index])); // 不要这样做!
return ListTile(...);
},
);
// ✅ 好:为每个项目使用独立的小部件
class TodoItem extends ConsumerWidget {
const TodoItem({required this.todoId});
final String todoId;
@override
Widget build(BuildContext context, WidgetRef ref) {
final todo = ref.watch(todoProvider(todoId));
return ListTile(title: Text(todo.title));
}
}
class TodoList extends ConsumerWidget {
@override
Widget build(BuildContext context, WidgetRef ref) {
final ids = ref.watch(todoIdsProvider);
return ListView.builder(
itemCount: ids.length,
itemBuilder: (context, index) => TodoItem(todoId: ids[index]),
);
}
}
// ✅ 好:为计算状态使用独立的提供者
@riverpod
List<Todo> filteredSortedTodos(Ref ref) {
final todos = ref.watch(todoListProvider);
final filter = ref.watch(filterProvider);
final sortOrder = ref.watch(sortOrderProvider);
final filtered = todos.where((t) => t.matches(filter)).toList();
return filtered..sort(sortOrder.comparator);
}
1. 数据层 - 仓库:
@riverpod
TodoRepository todoRepository(Ref ref) {
return TodoRepository(dio: ref.watch(dioProvider));
}
class TodoRepository {
TodoRepository({required this.dio});
final Dio dio;
Future<List<Todo>> fetchTodos() async {
final response = await dio.get('/todos');
return (response.data as List)
.map((json) => Todo.fromJson(json))
.toList();
}
Future<Todo> createTodo(String title) async {
final response = await dio.post('/todos', data: {'title': title});
return Todo.fromJson(response.data);
}
Future<void> deleteTodo(String id) async {
await dio.delete('/todos/$id');
}
}
2. 应用层 - 状态管理:
@riverpod
class TodoList extends _$TodoList {
@override
Future<List<Todo>> build() async {
final repository = ref.watch(todoRepositoryProvider);
return repository.fetchTodos();
}
Future<void> addTodo(String title) async {
final repository = ref.read(todoRepositoryProvider);
state = const AsyncLoading();
state = await AsyncValue.guard(() async {
await repository.createTodo(title);
return repository.fetchTodos();
});
}
}
3. 表示层 - 用户界面:
class TodoListScreen extends ConsumerWidget {
@override
Widget build(BuildContext context, WidgetRef ref) {
final todosAsync = ref.watch(todoListProvider);
return Scaffold(
appBar: AppBar(title: const Text('Todos')),
body: todosAsync.when(
data: (todos) => ListView.builder(
itemCount: todos.length,
itemBuilder: (context, index) => TodoTile(todos[index]),
),
loading: () => const Center(child: CircularProgressIndicator()),
error: (error, stack) => ErrorView(error: error),
),
floatingActionButton: FloatingActionButton(
onPressed: () => _showAddDialog(context, ref),
child: const Icon(Icons.add),
),
);
}
}
// 服务
@riverpod
Dio dio(Ref ref) {
final dio = Dio(BaseOptions(baseUrl: 'https://api.example.com'));
dio.interceptors.add(LogInterceptor());
return dio;
}
@riverpod
Future<SharedPreferences> sharedPreferences(Ref ref) async {
return await SharedPreferences.getInstance();
}
@riverpod
AuthService authService(Ref ref) {
return AuthService(
dio: ref.watch(dioProvider),
storage: ref.watch(sharedPreferencesProvider).value!,
);
}
// 仓库依赖于服务
@riverpod
UserRepository userRepository(Ref ref) {
return UserRepository(
dio: ref.watch(dioProvider),
authService: ref.watch(authServiceProvider),
);
}
// 状态提供者依赖于仓库
@riverpod
class CurrentUser extends _$CurrentUser {
@override
Future<User?> build() async {
final authService = ref.watch(authServiceProvider);
final userId = await authService.getCurrentUserId();
if (userId == null) return null;
final repository = ref.watch(userRepositoryProvider);
return repository.fetchUser(userId);
}
Future<void> logout() async {
final authService = ref.read(authServiceProvider);
await authService.logout();
ref.invalidateSelf();
}
}
当您添加参数时,代码生成会自动创建家族提供者:
// 简单的家族提供者
@riverpod
Future<User> user(Ref ref, String id) async {
final dio = ref.watch(dioProvider);
final response = await dio.get('/users/$id');
return User.fromJson(response.data);
}
// 用法
final user = ref.watch(userProvider('123'));
// 使用 AsyncNotifier 的家族提供者
@riverpod
class UserNotifier extends _$UserNotifier {
@override
Future<User> build(String id) async {
final repo = ref.watch(userRepositoryProvider);
return repo.fetchUser(id);
}
Future<void> updateName(String newName) async {
final userId = arg; // 通过 'arg' 访问参数
state = const AsyncLoading();
state = await AsyncValue.guard(() async {
await ref.read(userRepositoryProvider).updateUser(userId, name: newName);
return ref.read(userRepositoryProvider).fetchUser(userId);
});
}
}
// 复杂参数需要正确的相等性判断
class UserFilter {
const UserFilter({required this.role, required this.active});
final String role;
final bool active;
@override
bool operator ==(Object other) =>
identical(this, other) ||
other is UserFilter &&
role == other.role &&
active == other.active;
@override
int get hashCode => Object.hash(role, active);
}
@riverpod
Future<List<User>> filteredUsers(Ref ref, UserFilter filter) async {
return fetchUsers(filter);
}
代码生成使提供者默认启用自动释放。
// 默认:当没有监听器时自动释放
@riverpod
Future<String> data(Ref ref) async => fetchData();
// 永久保持活动状态
@Riverpod(keepAlive: true)
Future<Config> config(Ref ref) async => loadConfig();
// 条件性保持活动状态 - 成功时缓存
@riverpod
Future<String> cachedData(Ref ref) async {
final data = await fetchData();
ref.keepAlive(); // 永久缓存此结果
return data;
}
// 定时缓存 (5 分钟)
@riverpod
Future<String> timedCache(Ref ref) async {
final data = await fetchData();
final link = ref.keepAlive();
Timer(const Duration(minutes: 5), link.close);
return data;
}
// 手动释放 - 清理资源
@riverpod
Stream<int> websocket(Ref ref) {
final client = WebSocketClient();
ref.onDispose(() {
client.close(); // 当提供者被释放时清理
});
return client.stream;
}
@riverpod
class TodoList extends _$TodoList {
@override
Future<List<Todo>> build() async {
try {
final repository = ref.watch(todoRepositoryProvider);
return await repository.fetchTodos();
} on DioException catch (e) {
if (e.response?.statusCode == 401) {
ref.read(authServiceProvider).logout();
throw UnauthorizedException();
}
throw NetworkException(e.message);
} catch (e) {
throw UnexpectedException(e.toString());
}
}
Future<void> addTodo(String title) async {
state = const AsyncLoading();
state = await AsyncValue.guard(() async {
final repository = ref.read(todoRepositoryProvider);
await repository.createTodo(title);
return repository.fetchTodos();
});
}
}
// 使用 .when()
todosAsync.when(
data: (todos) => ListView.builder(...),
loading: () => const Center(child: CircularProgressIndicator()),
error: (error, stack) {
if (error is NetworkException) {
return ErrorView(
message: '网络错误。请检查您的连接。',
onRetry: () => ref.invalidate(todoListProvider),
);
}
if (error is UnauthorizedException) {
return const ErrorView(message: '请重新登录。');
}
return ErrorView(message: '错误: $error');
},
);
// 监听错误 (副作用)
ref.listen<AsyncValue<List<Todo>>>(
todoListProvider,
(previous, next) {
next.whenOrNull(
error: (error, stack) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(
content: Text(error.toString()),
backgroundColor: Colors.red,
),
);
},
);
},
);
test('TodoList fetches todos correctly', () async {
final container = ProviderContainer.test(
overrides: [
todoRepositoryProvider.overrideWithValue(MockTodoRepository()),
],
);
final todos = await container.read(todoListProvider.future);
expect(todos.length, 2);
expect(todos[0].title, 'Test Todo 1');
});
test('TodoList adds todo correctly', () async {
final mockRepo = MockTodoRepository();
final container = ProviderContainer.test(
overrides: [
todoRepositoryProvider.overrideWithValue(mockRepo),
],
);
await container.read(todoListProvider.notifier).addTodo('New Todo');
verify(() => mockRepo.createTodo('New Todo')).called(1);
});
testWidgets('TodoListScreen displays todos', (tester) async {
await tester.pumpWidget(
ProviderScope(
overrides: [
todoRepositoryProvider.overrideWithValue(MockTodoRepository()),
],
child: const MaterialApp(home: TodoListScreen()),
),
);
await tester.pumpAndSettle();
expect(find.text('Test Todo 1'), findsOneWidget);
expect(find.text('Test Todo 2'), findsOneWidget);
});
// ❌ 使用 ref.read() 来避免重建
final todos = ref.read(todoListProvider); // 不会重建!
// ✅ 使用 ref.watch() 或 ref.select()
final count = ref.watch(todoListProvider.select((todos) => todos.length));
// ❌ 不释放资源
@riverpod
Stream<int> badWebsocket(Ref ref) {
final client = WebSocketClient();
return client.stream; // 永远不会关闭!
}
// ✅ 释放资源
@riverpod
Stream<int> goodWebsocket(Ref ref) {
final client = WebSocketClient();
ref.onDispose(() => client.close());
return client.stream;
}
// ❌ 不好:哪个是数据源?
class BadWidget extends StatefulWidget {
int localCount = 0; // 本地状态
@override
Widget build(BuildContext context, WidgetRef ref) {
final providerCount = ref.watch(counterProvider); // 提供者状态
return Text('$localCount vs $providerCount'); // 令人困惑!
}
}
// ✅ 好:单一数据源
class GoodWidget extends ConsumerWidget {
@override
Widget build(BuildContext context, WidgetRef ref) {
final count = ref.watch(counterProvider);
return Text('$count');
}
}
// ❌ 不好
Future<void> logout() async {
state = null;
// 其他提供者仍然持有旧的用户数据!
}
// ✅ 好:使依赖的提供者失效
Future<void> logout() async {
state = null;
ref.invalidate(userProfileProvider);
ref.invalidate(userSettingsProvider);
ref.invalidate(userNotificationsProvider);
}
// ✅ 更好:让提供者监听认证状态
@riverpod
Future<UserProfile> userProfile(Ref ref) async {
final user = ref.watch(authProvider);
if (user == null) throw UnauthenticatedException();
return fetchUserProfile(user.id); // 当用户变化时自动重新获取
}
当用户使用 Riverpod 时:
@riverpod 注解select() 来优化性能,当监听特定字段时有关完整详情和高级模式,请参阅:/Users/pablito/EVOworkspace/flutter/CesarferPromotoresFlutter/promotores/RIVERPOD_2025_BEST_PRACTICES.md
每周安装次数
399
仓库
首次出现
2026 年 1 月 26 日
安全审计
安装于
codex320
opencode304
gemini-cli300
github-copilot287
kimi-cli261
amp259
Immutable/Computed Values - Use Provider:
@riverpod
String apiKey(Ref ref) => 'YOUR_API_KEY';
@riverpod
int totalPrice(Ref ref) {
final cart = ref.watch(cartProvider);
return cart.items.fold(0, (sum, item) => sum + item.price);
}
Simple Synchronous State - Use NotifierProvider:
@riverpod
class Counter extends _$Counter {
@override
int build() => 0;
void increment() => state++;
void decrement() => state = max(0, state - 1);
}
Async Data with Mutations (PREFERRED 2025) - Use AsyncNotifierProvider:
@riverpod
class TodoList extends _$TodoList {
@override
Future<List<Todo>> build() async {
final repo = ref.watch(todoRepositoryProvider);
return repo.fetchTodos();
}
Future<void> addTodo(String title) async {
state = const AsyncLoading();
state = await AsyncValue.guard(() async {
final repo = ref.read(todoRepositoryProvider);
await repo.createTodo(title);
return repo.fetchTodos();
});
}
Future<void> deleteTodo(String id) async {
// Optimistic update
state = AsyncData(state.value!.where((t) => t.id != id).toList());
try {
await ref.read(todoRepositoryProvider).deleteTodo(id);
} catch (e) {
ref.invalidateSelf(); // Rollback on error
}
}
}
Real-time Streams Only - Use StreamProvider:
@riverpod
Stream<User?> authState(Ref ref) {
return FirebaseAuth.instance.authStateChanges();
}
Key Rule : Prefer AsyncNotifierProvider over FutureProvider/StreamProvider for better consistency and mutation support.
dependencies:
flutter_riverpod: ^2.5.0
riverpod_annotation: ^2.3.0
dev_dependencies:
build_runner: ^2.4.0
riverpod_generator: ^2.4.0
custom_lint: ^0.6.0
riverpod_lint: ^2.3.0
Every provider file needs:
import 'package:riverpod_annotation/riverpod_annotation.dart';
part 'filename.g.dart'; // REQUIRED
@riverpod
class MyProvider extends _$MyProvider {
@override
Future<Data> build() async => fetchData();
}
# Watch mode (RECOMMENDED during development)
dart run build_runner watch -d
# One-time generation
dart run build_runner build --delete-conflicting-outputs
// ❌ BAD: Rebuilds on ANY product change
final product = ref.watch(productProvider);
return Text('\$${product.price}');
// ✅ GOOD: Only rebuilds when price changes
final price = ref.watch(productProvider.select((p) => p.price));
return Text('\$$price');
ref.watch() - Subscribe to changes (use in build):
@override
Widget build(BuildContext context, WidgetRef ref) {
final todos = ref.watch(todoListProvider);
return ListView(...);
}
ref.select() - Subscribe to specific property (optimize rebuilds):
final count = ref.watch(todoListProvider.select((todos) => todos.length));
final isAdult = ref.watch(personProvider.select((p) => p.age >= 18));
ref.read() - One-time read with NO subscription (event handlers only):
onPressed: () {
ref.read(todoListProvider.notifier).addTodo('New task');
}
// ❌ NEVER use read() in build to "optimize" - it won't rebuild!
ref.listen() - Side effects (navigation, snackbars, logging):
ref.listen<AsyncValue<List<Todo>>>(
todoListProvider,
(previous, next) {
next.whenOrNull(
error: (error, stack) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(content: Text('Error: $error')),
);
},
);
},
);
// ❌ BAD: Causes performance issues
ListView.builder(
itemBuilder: (context, index) {
final todo = ref.watch(todoProvider(ids[index])); // DON'T!
return ListTile(...);
},
);
// ✅ GOOD: Separate widget for each item
class TodoItem extends ConsumerWidget {
const TodoItem({required this.todoId});
final String todoId;
@override
Widget build(BuildContext context, WidgetRef ref) {
final todo = ref.watch(todoProvider(todoId));
return ListTile(title: Text(todo.title));
}
}
class TodoList extends ConsumerWidget {
@override
Widget build(BuildContext context, WidgetRef ref) {
final ids = ref.watch(todoIdsProvider);
return ListView.builder(
itemCount: ids.length,
itemBuilder: (context, index) => TodoItem(todoId: ids[index]),
);
}
}
// ✅ GOOD: Separate provider for computed state
@riverpod
List<Todo> filteredSortedTodos(Ref ref) {
final todos = ref.watch(todoListProvider);
final filter = ref.watch(filterProvider);
final sortOrder = ref.watch(sortOrderProvider);
final filtered = todos.where((t) => t.matches(filter)).toList();
return filtered..sort(sortOrder.comparator);
}
1. Data Layer - Repository :
@riverpod
TodoRepository todoRepository(Ref ref) {
return TodoRepository(dio: ref.watch(dioProvider));
}
class TodoRepository {
TodoRepository({required this.dio});
final Dio dio;
Future<List<Todo>> fetchTodos() async {
final response = await dio.get('/todos');
return (response.data as List)
.map((json) => Todo.fromJson(json))
.toList();
}
Future<Todo> createTodo(String title) async {
final response = await dio.post('/todos', data: {'title': title});
return Todo.fromJson(response.data);
}
Future<void> deleteTodo(String id) async {
await dio.delete('/todos/$id');
}
}
2. Application Layer - State Management :
@riverpod
class TodoList extends _$TodoList {
@override
Future<List<Todo>> build() async {
final repository = ref.watch(todoRepositoryProvider);
return repository.fetchTodos();
}
Future<void> addTodo(String title) async {
final repository = ref.read(todoRepositoryProvider);
state = const AsyncLoading();
state = await AsyncValue.guard(() async {
await repository.createTodo(title);
return repository.fetchTodos();
});
}
}
3. Presentation Layer - UI :
class TodoListScreen extends ConsumerWidget {
@override
Widget build(BuildContext context, WidgetRef ref) {
final todosAsync = ref.watch(todoListProvider);
return Scaffold(
appBar: AppBar(title: const Text('Todos')),
body: todosAsync.when(
data: (todos) => ListView.builder(
itemCount: todos.length,
itemBuilder: (context, index) => TodoTile(todos[index]),
),
loading: () => const Center(child: CircularProgressIndicator()),
error: (error, stack) => ErrorView(error: error),
),
floatingActionButton: FloatingActionButton(
onPressed: () => _showAddDialog(context, ref),
child: const Icon(Icons.add),
),
);
}
}
// Services
@riverpod
Dio dio(Ref ref) {
final dio = Dio(BaseOptions(baseUrl: 'https://api.example.com'));
dio.interceptors.add(LogInterceptor());
return dio;
}
@riverpod
Future<SharedPreferences> sharedPreferences(Ref ref) async {
return await SharedPreferences.getInstance();
}
@riverpod
AuthService authService(Ref ref) {
return AuthService(
dio: ref.watch(dioProvider),
storage: ref.watch(sharedPreferencesProvider).value!,
);
}
// Repositories depend on services
@riverpod
UserRepository userRepository(Ref ref) {
return UserRepository(
dio: ref.watch(dioProvider),
authService: ref.watch(authServiceProvider),
);
}
// State providers depend on repositories
@riverpod
class CurrentUser extends _$CurrentUser {
@override
Future<User?> build() async {
final authService = ref.watch(authServiceProvider);
final userId = await authService.getCurrentUserId();
if (userId == null) return null;
final repository = ref.watch(userRepositoryProvider);
return repository.fetchUser(userId);
}
Future<void> logout() async {
final authService = ref.read(authServiceProvider);
await authService.logout();
ref.invalidateSelf();
}
}
Family providers are automatic with code generation when you add parameters:
// Simple family provider
@riverpod
Future<User> user(Ref ref, String id) async {
final dio = ref.watch(dioProvider);
final response = await dio.get('/users/$id');
return User.fromJson(response.data);
}
// Usage
final user = ref.watch(userProvider('123'));
// Family with AsyncNotifier
@riverpod
class UserNotifier extends _$UserNotifier {
@override
Future<User> build(String id) async {
final repo = ref.watch(userRepositoryProvider);
return repo.fetchUser(id);
}
Future<void> updateName(String newName) async {
final userId = arg; // Access the parameter via 'arg'
state = const AsyncLoading();
state = await AsyncValue.guard(() async {
await ref.read(userRepositoryProvider).updateUser(userId, name: newName);
return ref.read(userRepositoryProvider).fetchUser(userId);
});
}
}
// Complex parameters need proper equality
class UserFilter {
const UserFilter({required this.role, required this.active});
final String role;
final bool active;
@override
bool operator ==(Object other) =>
identical(this, other) ||
other is UserFilter &&
role == other.role &&
active == other.active;
@override
int get hashCode => Object.hash(role, active);
}
@riverpod
Future<List<User>> filteredUsers(Ref ref, UserFilter filter) async {
return fetchUsers(filter);
}
Code generation makes providers auto-dispose by default.
// Default: auto-dispose when no listeners
@riverpod
Future<String> data(Ref ref) async => fetchData();
// Keep alive permanently
@Riverpod(keepAlive: true)
Future<Config> config(Ref ref) async => loadConfig();
// Conditional keep alive - cache on success
@riverpod
Future<String> cachedData(Ref ref) async {
final data = await fetchData();
ref.keepAlive(); // Cache this result forever
return data;
}
// Timed cache (5 minutes)
@riverpod
Future<String> timedCache(Ref ref) async {
final data = await fetchData();
final link = ref.keepAlive();
Timer(const Duration(minutes: 5), link.close);
return data;
}
// Manual disposal - cleanup resources
@riverpod
Stream<int> websocket(Ref ref) {
final client = WebSocketClient();
ref.onDispose(() {
client.close(); // Cleanup when provider is disposed
});
return client.stream;
}
@riverpod
class TodoList extends _$TodoList {
@override
Future<List<Todo>> build() async {
try {
final repository = ref.watch(todoRepositoryProvider);
return await repository.fetchTodos();
} on DioException catch (e) {
if (e.response?.statusCode == 401) {
ref.read(authServiceProvider).logout();
throw UnauthorizedException();
}
throw NetworkException(e.message);
} catch (e) {
throw UnexpectedException(e.toString());
}
}
Future<void> addTodo(String title) async {
state = const AsyncLoading();
state = await AsyncValue.guard(() async {
final repository = ref.read(todoRepositoryProvider);
await repository.createTodo(title);
return repository.fetchTodos();
});
}
}
// Using .when()
todosAsync.when(
data: (todos) => ListView.builder(...),
loading: () => const Center(child: CircularProgressIndicator()),
error: (error, stack) {
if (error is NetworkException) {
return ErrorView(
message: 'Network error. Check your connection.',
onRetry: () => ref.invalidate(todoListProvider),
);
}
if (error is UnauthorizedException) {
return const ErrorView(message: 'Please log in again.');
}
return ErrorView(message: 'Error: $error');
},
);
// Listen for errors (side effects)
ref.listen<AsyncValue<List<Todo>>>(
todoListProvider,
(previous, next) {
next.whenOrNull(
error: (error, stack) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(
content: Text(error.toString()),
backgroundColor: Colors.red,
),
);
},
);
},
);
test('TodoList fetches todos correctly', () async {
final container = ProviderContainer.test(
overrides: [
todoRepositoryProvider.overrideWithValue(MockTodoRepository()),
],
);
final todos = await container.read(todoListProvider.future);
expect(todos.length, 2);
expect(todos[0].title, 'Test Todo 1');
});
test('TodoList adds todo correctly', () async {
final mockRepo = MockTodoRepository();
final container = ProviderContainer.test(
overrides: [
todoRepositoryProvider.overrideWithValue(mockRepo),
],
);
await container.read(todoListProvider.notifier).addTodo('New Todo');
verify(() => mockRepo.createTodo('New Todo')).called(1);
});
testWidgets('TodoListScreen displays todos', (tester) async {
await tester.pumpWidget(
ProviderScope(
overrides: [
todoRepositoryProvider.overrideWithValue(MockTodoRepository()),
],
child: const MaterialApp(home: TodoListScreen()),
),
);
await tester.pumpAndSettle();
expect(find.text('Test Todo 1'), findsOneWidget);
expect(find.text('Test Todo 2'), findsOneWidget);
});
// ❌ Using ref.read() to avoid rebuilds
final todos = ref.read(todoListProvider); // Won't rebuild!
// ✅ Use ref.watch() or ref.select()
final count = ref.watch(todoListProvider.select((todos) => todos.length));
// ❌ Not disposing resources
@riverpod
Stream<int> badWebsocket(Ref ref) {
final client = WebSocketClient();
return client.stream; // Never closed!
}
// ✅ Dispose resources
@riverpod
Stream<int> goodWebsocket(Ref ref) {
final client = WebSocketClient();
ref.onDispose(() => client.close());
return client.stream;
}
// ❌ BAD: Which is the source of truth?
class BadWidget extends StatefulWidget {
int localCount = 0; // Local state
@override
Widget build(BuildContext context, WidgetRef ref) {
final providerCount = ref.watch(counterProvider); // Provider state
return Text('$localCount vs $providerCount'); // Confusing!
}
}
// ✅ GOOD: Single source of truth
class GoodWidget extends ConsumerWidget {
@override
Widget build(BuildContext context, WidgetRef ref) {
final count = ref.watch(counterProvider);
return Text('$count');
}
}
// ❌ BAD
Future<void> logout() async {
state = null;
// Other providers still have old user data!
}
// ✅ GOOD: Invalidate dependent providers
Future<void> logout() async {
state = null;
ref.invalidate(userProfileProvider);
ref.invalidate(userSettingsProvider);
ref.invalidate(userNotificationsProvider);
}
// ✅ EVEN BETTER: Make providers watch auth
@riverpod
Future<UserProfile> userProfile(Ref ref) async {
final user = ref.watch(authProvider);
if (user == null) throw UnauthenticatedException();
return fetchUserProfile(user.id); // Auto-refetches when user changes
}
When the user is working with Riverpod:
@riverpod annotationsselect() when watching specific fieldsFor complete details and advanced patterns, refer to: /Users/pablito/EVOworkspace/flutter/CesarferPromotoresFlutter/promotores/RIVERPOD_2025_BEST_PRACTICES.md
Weekly Installs
399
Repository
First Seen
Jan 26, 2026
Security Audits
Installed on
codex320
opencode304
gemini-cli300
github-copilot287
kimi-cli261
amp259
Flutter 主屏幕小组件开发指南:iOS/Android 原生小组件集成与数据通信
2,200 周安装