flutter - BLoC - 每个州都应该是一个单独的类(class)吗?

标签 flutter state bloc state-management

我正在使用 BLoC 模式学习和构建 Flutter 应用程序,在很多教程和存储库中,我看到人们为 BLoC 的每个状态都有一个单独的类,而其他人有一个包含所有属性的单一状态类定义。 是否有用于定义状态类的标准 BLoC 方式,还是个人选择?

具有多个状态类的示例

abstract class LoginState extends Equatable {
  LoginState([List props = const []]) : super(props);
}

class LoginInitial extends LoginState {
  @override
  String toString() => 'LoginInitial';
}

class LoginLoading extends LoginState {
  @override
  String toString() => 'LoginLoading';
}

class LoginFailure extends LoginState {
  final String error;

  LoginFailure({@required this.error}) : super([error]);

  @override
  String toString() => 'LoginFailure { error: $error }';
}

单个状态类的示例

@immutable
class MyFormState extends Equatable {
  final String email;
  final bool isEmailValid;
  final String password;
  final bool isPasswordValid;
  final bool formSubmittedSuccessfully;

  bool get isFormValid => isEmailValid && isPasswordValid;

  MyFormState({
    @required this.email,
    @required this.isEmailValid,
    @required this.password,
    @required this.isPasswordValid,
    @required this.formSubmittedSuccessfully,
  }) : super([
          email,
          isEmailValid,
          password,
          isPasswordValid,
          formSubmittedSuccessfully,
        ]);

  factory MyFormState.initial() {
    return MyFormState(
      email: '',
      isEmailValid: false,
      password: '',
      isPasswordValid: false,
      formSubmittedSuccessfully: false,
    );
  }

  MyFormState copyWith({
    String email,
    bool isEmailValid,
    String password,
    bool isPasswordValid,
    bool formSubmittedSuccessfully,
  }) {
    return MyFormState(
      email: email ?? this.email,
      isEmailValid: isEmailValid ?? this.isEmailValid,
      password: password ?? this.password,
      isPasswordValid: isPasswordValid ?? this.isPasswordValid,
      formSubmittedSuccessfully:
          formSubmittedSuccessfully ?? this.formSubmittedSuccessfully,
    );
  }

  @override
  String toString() {
    return '''MyFormState {
      email: $email,
      isEmailValid: $isEmailValid,
      password: $password,
      isPasswordValid: $isPasswordValid,
      formSubmittedSuccessfully: $formSubmittedSuccessfully
    }''';
  }
}

什么时候应该使用哪一个? 两者有什么优缺点?

最佳答案

这更多是个人选择/编码风格,但我同意根据情况,某些方式可能比其他方式更好。

请注意 Bloc doc现在说明这两种方法。

虽然有一些差异。首先,当使用多个状态类时,您可以更好地控制每个状态的字段。例如,如果您知道错误只会在某些情况下发生,最好将其隔离在特定类 BlocSubjectError 中,而不是让一个类包含可能存在或可能不存在的可空错误。 从调用者的角度来看,条件也有点不同,您可能更喜欢一种方式:

具有多个状态类

if(state is BlocSubjectLoading) {
  // display loading indicator
}
else if (state is BlocSubjectError) {
  // display errror
} else if (state is BlocSubjectSuccess) {
  // display data
}

单状态类

if(state.isLoading) {
  // display loading indicator
}
else if (state.error != null) {
  // display errror
} else if (state.data != null) {
  // display data
}

虽然看起来可能没有太大区别,但第二个示例允许混合状态:您可能同时有错误数据,或者两者都在加载有一个以前的数据。它可能是也可能是你想要的。在第一种情况下,类型使其更加受限,这使得该技术更接近于用类型表示应用程序的状态,也就是 Type Driven Development。 .

另一方面,如果它们都声明相同的字段,那么拥有多个状态类可能会很麻烦,因为您每次都需要使用所有参数实例化您的状态,而使用单个类您只需调用一个 。 copyWith() 方法,您将只向其提供更改的参数。

单状态类模式接近Triple Pattern (或分段状态模式)。您可以在 Bloc here 中找到描述此模式的综合研讨会。 .

关于flutter - BLoC - 每个州都应该是一个单独的类(class)吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58144116/

相关文章:

flutter - 使用 RefreshIndicator 时,无法将参数类型 'Future<List<User>>' 分配给 onRefresh 中的参数类型 'Future<void> Function()'

javascript - 设置状态后设置状态

android - 如何在flutter中将token设置为全局变量?所以我可以随时随地访问 token

有状态小部件的 ListView

android - 从 Flutter 应用程序中的 QuerySnapshot 变量中提取信息

flutter - 如何将字符插入 Dart 字符串?

flutter - CupertinoPageScaffold 的子小部件位于 CupertinoNavigationBar 后面

javascript - 我正在尝试从最低级别组件上的输入字段捕获数据

javascript - 无法在其他函数中访问 this.state

firebase - Flutter 状态管理问题