Riverpod 的 StreamProvider 只产生一次 StreamValue |颤动和蜂巢

问题描述

我编写了一个 StreamProvider,我在启动后立即收听它,以获取有关潜在登录用户的所有信息。如果没有用户,那么结果将为空,监听器保持加载状态,所以我决定发回一个用户认值,让我知道加载完成。 我必须这样做,因为 Hive 的 watch() 方法仅在数据更改时触发,而不会在启动时触发。 所以在那之后,我希望 watch() 方法完成它的工作,但问题在于以下情况:

  1. 启动时:无用户 - 插入用户 -> 触发 watch 方法 -> 我获取插入的用户数据 -> 删除登录用户 -> 未触发 watch 方法.

  2. 启动时:完整用户 - 删除用户 -> watch 方法被触发 -> 我得到一个用户 -> 插入一个用户 -> watch 方法没有被触发。 >

一段时间后,我发现我可以随心所欲地使用所有的 CRUD 操作,并且 Hive 的盒子做了它应该做的事情,但是 watch() 方法在它被触发后不再被触发 一次

流提供者:

final localUsersBoxFutureProvider = FutureProvider<Box>((ref) async {
  final usersBox = await Hive.openBox('users');
  return usersBox;
});

final localUserStreamProvider = StreamProvider<User>((ref) async* {
  final usersBox = await ref.watch(localUsersBoxFutureProvider.future);

  yield* Stream.value(usersBox.get(0,defaultValue: User()));
  yield* usersBox.watch(key: 0).map((usersBoxEvent) {
    return usersBoxEvent.value == null ? User() : usersBoxEvent.value as User;
  });
});

听众:

return localUserStream.when(
  data: (data) {
    if (data.name == null) {
      print('Emitted data is an empty user');
    } else {
      print('Emitted data is a full user');
    }

    return Container(color: Colors.blue,child: Center(child: Row(children: [
      RawMaterialButton(
        onpressed: () async {
          final globalResponse = await globalDatabaseService.signup({
            'email' : 'name@email.com','password' : 'password','name' : 'My Name'
          });

          Map<String,dynamic> jsonString = jsonDecode(globalResponse.bodyString);
          await localDatabaseService.insertUser(User.fromJSON(jsonString));
        },child: Text('Insert'),),RawMaterialButton(
        onpressed: () async {
          await localDatabaseService.removeUser();
        },child: Text('Delete'),)
    ])));
  },loading: () {
    return Container(color: Colors.yellow);
  },error: (e,s) {
    return Container(color: Colors.red);
  }
);

CRUD 方法

Future<void> insertUser(User user) async {
    Box usersBox = await Hive.openBox('users');
    await usersBox.put(0,user);
    await usersBox.close();
  }

  Future<User> readUser() async {
    Box usersBox = await Hive.openBox('users');
    User user = usersBox.get(0) as User;
    await usersBox.close();
    return user;
  }

  Future<void> removeUser() async {
    Box usersBox = await Hive.openBox('users');
    await usersBox.delete(0);
    await usersBox.close();
  }

知道如何告诉 StreamProvider watch() 方法应该保持活动状态,即使已经发出了一个值?

解决方法

但是 watch() 方法被触发后不再被触发 一次

那是因为在每个 CRUD 之后你都关闭了框,所以流(使用那个框)停止发出值。如果您从 riverpod (await Hive.openBox('users')) 之外的某个地方调用它,它会调用相同的引用,这无关紧要。你应该只在你停止使用它时关闭它,我建议使用带有 riverpod 的 autodispose 在不再使用时关闭它,并且可能将那些 CRUD 方法放在一个由 riverpod 控制的类中,这样你就可以完全控制它的生命周期盒子

final localUsersBoxFutureProvider = FutureProvider.autoDispose<Box>((ref) async {
  final usersBox = await Hive.openBox('users');

  ref.onDispose(() async => await usersBox?.close()); //this will close the box automatically when the provider is no longer used
  return usersBox;
});

final localUserStreamProvider = StreamProvider<User>.autoDispose((ref) async* {
  final usersBox = await ref.watch(localUsersBoxFutureProvider.future);

  yield* Stream.value(usersBox.get(0,defaultValue: User()));
  yield* usersBox.watch(key: 0).map((usersBoxEvent) {
    return usersBoxEvent.value == null ? User() : usersBoxEvent.value as User;
  });
});

并且在您的方法中使用来自 localUsersBoxFutureProvider 的相同实例框,并且不要在每个框之后关闭框,当您停止收听 streamlocalUsersBoxFutureProvider 时,它会自行关闭