Flutter/Package

[Flutter] Bloc ④ - context.read, context.watch, context.select

찌김이 2022. 8. 5. 15:01
728x90
반응형

Bloc ① - 소개  https://dalgoodori.tistory.com/28

Bloc ② - Cubit 과 Bloc https://dalgoodori.tistory.com/29

Bloc ③ - Bloc Widgets https://dalgoodori.tistory.com/30

Bloc  - context.read, context.watch, context.select

Bloc ⑤ - Bloc Test https://dalgoodori.tistory.com/32

 

1. context.read<T>

  • context.read<T>()는 T 유형의 가장 가까운 조상 인스턴스를 찾습니다.
  • 기능적으로 BlocProvider.of<T>(context)와 동일합니다. 
  • context.read는 onPressed 콜백 내에 이벤트를 추가하기 위해 블록 인스턴스를 검색하는 데 가장 일반적으로 사용됩니다.
onPressed() {
  context.read<CounterBloc>().add(CounterIncrementPressed()),
}

 

  • 사용시 주의할 점은 빌드 메소드 내에서 상태를 검색하기 위해 context.read를 사용하면 안됩니다.  
  • 블록의 상태가 변경되면 위젯이 다시 빌드되지 않기 때문에 오류가 발생할 수 있습니다.
@override
Widget build(BuildContext context) {
  final state = context.read<MyBloc>().state;
  return Text('$state');
}

 

2. context.watch<T>

  • context.watch<T>()는 T 유형의 가장 가까운 조상 인스턴스를 제공하지만 인스턴스의 변경 사항도 수신합니다. 
  • 기능적으로 BlocProvider.of<T>(context, listen: true)와 동일합니다.
  • 제공된 T 유형의 개체가 변경되면 context.watch가 다시 빌드를 트리거합니다.
  • context.watch는 StatelessWidget 또는 State 클래스의 빌드 메서드 내에서만 액세스할 수 있습니다.
  • Builder 위젯을 통해 BlocBuilder 대신 사용해도 됩니다.
  • Builder 위젯을 통해 MultiBlocBuilder 도 구현 가능합니다.
// BlocBuilder

@override
Widget build(BuildContext context) {
  return MaterialApp(
    home: Scaffold(
      body: BlocBuilder<MyBloc, MyState>(
        builder: (context, state) {
          // Whenever the state changes, only the Text is rebuilt.
          return Text(state.value);
        },
      ),
    ),
  );
}


// BlocBuilder 대신 Builder 로 구현

@override
Widget build(BuildContext context) {
  return MaterialApp(
    home: Scaffold(
      body: Builder(
        builder: (context) {
          // Whenever the state changes, only the Text is rebuilt.
          final state = context.watch<MyBloc>().state;
          return Text(state.value);
        },
      ),
    ),
  );
}


// Builder 를 통해 구현한 MultiBlocBuilder

@override
Widget build(BuildContext context) {
  return MaterialApp(
    home: Scaffold(
      body: Builder(
        builder: (context) {
          final stateA = context.watch<BlocA>().state;
          final stateB = context.watch<BlocB>().state;
          final stateC = context.watch<BlocC>().state;

          // return a Widget which depends on the state of BlocA, BlocB, and BlocC
          }
      ),
    ),
  );
}

 

  • 사용 시 주의할 점은 빌드 메소드의 상위 위젯이 상태에 의존하지 않을 때 context.watch를 사용하면 안됩니다.
  • build 메소드의 루트에서 context.watch를 사용하면 블록 상태가 변경될 때 전체 위젯이 다시 빌드되기 때문입니다.
@override
Widget build(BuildContext context) {
  // Whenever the state changes, the MaterialApp is rebuilt
  // even though it is only used in the Text widget.
  
  final state = context.watch<MyBloc>().state;
  return MaterialApp(
    home: Scaffold(
      body: Text(state.value),
    ),
  );
}

 

3. context.select<T, R>

  • context.select<T, R>(R function(T value))는 T 유형의 가장 가까운 조상 인스턴스를 제공하고 T의 변경 사항을 수신합니다.
  • context.watch와 달리 context.select 상태의 더 작은 부분에서 변경 사항을 수신할 수 있습니다.  
Widget build(BuildContext context) {
  
  // ProfileBloc 상태의 속성 이름이 변경될 때만 위젯을 다시 빌드합니다.
  final name = context.select((ProfileBloc bloc) => bloc.state.name);
  return Text(name);
}

 

  • Builder 위젯을 통해 BlocSelector 대신 사용해도 됩니다.
// BlocSelector

@override
Widget build(BuildContext context) {
  return MaterialApp(
    home: Scaffold(
      body: BlocSelector<ProfileBloc, ProfileState, String>(
        selector: (state) => state.name,
        builder: (context, name) {
          // Whenever the state.name changes, only the Text is rebuilt.
          return Text(name);
        },
      ),
    ),
  );
}


// BlocSelector 대신 Builder 로 구현

@override
Widget build(BuildContext context) {
  return MaterialApp(
    home: Scaffold(
      body: Builder(
        builder: (context) {
          // Whenever state.name changes, only the Text is rebuilt.
          final name = context.select((ProfileBloc bloc) => bloc.state.name);
          return Text(name);
        },
      ),
    ),
  );
}

 

  • 사용 시 주의할 점은 context.watch 와 동일합니다.
@override
Widget build(BuildContext context) {
	
  // 빌드 메소드의 루트에서 사용하면 블록이 변경될 때 전체 위젯이 다시 빌드됩니다.  
  final name = context.select((ProfileBloc bloc) => bloc.state.name);
  return MaterialApp(
    home: Scaffold(
      body: Text(name),
    ),
  );
}
728x90
반응형