๊ด€๋ฆฌ ๋ฉ”๋‰ด

ruriruriya

[Flutter] ๊ธฐ๋ณธ ์ƒํƒœ ๊ด€๋ฆฌ (State Management) ๋ณธ๋ฌธ

๐Ÿ“ฑFlutter/Flutter Framework

[Flutter] ๊ธฐ๋ณธ ์ƒํƒœ ๊ด€๋ฆฌ (State Management)

๋ฃจ๋ฆฌ์•ผใ…‘ 2025. 2. 22. 21:52
๋ฐ˜์‘ํ˜•

Flutter์—์„œ ์ƒํƒœ ๊ด€๋ฆฌ๋Š” UI๊ฐ€ ๋ณ€ํ™”ํ•˜๋Š” ๋ฐ์ดํ„ฐ๋ฅผ ํšจ๊ณผ์ ์œผ๋กœ ์œ ์ง€ํ•˜๊ณ  ๋ฐ˜์˜ํ•˜๋Š” ๋ฐฉ๋ฒ•์ด๋‹ค. ์ƒํƒœ ๊ด€๋ฆฌ๊ฐ€ ์ค‘์š”ํ•œ ์ด์œ ๋Š” Flutter์˜ ์„ ์–ธ์  UI ๋ฐฉ์‹ ๋•Œ๋ฌธ์ด๋‹ค. ์ฆ‰, UI๋Š” ์ƒํƒœ(state)์— ๋”ฐ๋ผ ๋ณ€๊ฒฝ๋˜๋ฏ€๋กœ ์ด๋ฅผ ์–ด๋–ป๊ฒŒ ๋‹ค๋ฃจ๋Š๋ƒ์— ๋”ฐ๋ผ ์•ฑ์˜ ๊ตฌ์กฐ์™€ ์œ ์ง€๋ณด์ˆ˜์„ฑ์ด ๋‹ฌ๋ผ์ง„๋‹ค.

์—ฌ๊ธฐ์„œ๋Š” Flutter์˜ ๊ธฐ๋ณธ ์ƒํƒœ ๊ด€๋ฆฌ ๋ฐฉ๋ฒ•์— ๋Œ€ํ•ด ์„ค๋ช…ํ•œ๋‹ค. ์„œ๋“œํŒŒํ‹ฐ ๋ผ์ด๋ธŒ๋Ÿฌ๋ฆฌ(Provider, Riverpod ๋“ฑ)๋ฅผ ์‚ฌ์šฉํ•˜์ง€ ์•Š๊ณ  Flutter์—์„œ ์ œ๊ณตํ•˜๋Š” ๋ฐฉ๋ฒ•๋งŒ ๋‹ค๋ฃฌ๋‹ค.

1. ์ƒํƒœ(State)๋ž€?

Flutter์—์„œ ์ƒํƒœ๋ž€ ์•ฑ์˜ UI๋ฅผ ๊ฒฐ์ •ํ•˜๋Š” ๋ฐ์ดํ„ฐ์ด๋‹ค. ์ƒํƒœ๋Š” ํฌ๊ฒŒ ๋‘ ๊ฐ€์ง€๋กœ ๋‚˜๋‰œ๋‹ค:

  1. Ephemeral state (๋‹จ๊ธฐ ์ƒํƒœ): ํŠน์ • ์œ„์ ฏ ๋‚ด์—์„œ๋งŒ ๊ด€๋ฆฌ๋˜๋Š” ์ƒํƒœ (์˜ˆ: TextField ์ž…๋ ฅ๊ฐ’, ๋ฒ„ํŠผ ํด๋ฆญ ์—ฌ๋ถ€ ๋“ฑ)
  2. App state (์•ฑ ์ „์—ญ ์ƒํƒœ): ์—ฌ๋Ÿฌ ์œ„์ ฏ์ด ๊ณต์œ ํ•˜๋Š” ์ƒํƒœ (์˜ˆ: ์‚ฌ์šฉ์ž ์ธ์ฆ ์ •๋ณด, ํ…Œ๋งˆ ์„ค์ • ๋“ฑ)

 

2. StatefulWidget์„ ํ™œ์šฉํ•œ ์ƒํƒœ ๊ด€๋ฆฌ

๊ฐ€์žฅ ๊ธฐ๋ณธ์ด๊ณ  ๊ฐ„๋‹จํ•œ ์ƒํƒœ ๊ด€๋ฆฌ ๋ฐฉ๋ฒ•์€ StatefulWidget์„ ์ด์šฉํ•˜๋Š” ๊ฒƒ์ด๋‹ค.

StatefulWidget์€ ๋‚ด๋ถ€์ ์œผ๋กœ State ๊ฐ์ฒด๋ฅผ ๊ฐ€์ง€๋ฉฐ, setState() ๋ฉ”์„œ๋“œ๋ฅผ ์‚ฌ์šฉํ•˜์—ฌ ์ƒํƒœ๋ฅผ ๋ณ€๊ฒฝํ•  ์ˆ˜ ์žˆ๋‹ค. setState()๊ฐ€ ํ˜ธ์ถœ๋˜๋ฉด, build() ๋ฉ”์„œ๋“œ๊ฐ€ ๋‹ค์‹œ ์‹คํ–‰๋˜์–ด UI๊ฐ€ ๋ณ€๊ฒฝ๋œ ์ƒํƒœ๋ฅผ ๋ฐ˜์˜ํ•˜๊ฒŒ ๋œ๋‹ค.

StatefulWidget์˜ ํŠน์ง•

  • ์ž์ฒด์ ์œผ๋กœ ์ƒํƒœ๋ฅผ ๊ด€๋ฆฌํ•  ์ˆ˜ ์žˆ๋‹ค.
  • UI ๋ณ€๊ฒฝ์ด ํ•„์š”ํ•  ๋•Œ setState()๋ฅผ ํ˜ธ์ถœํ•˜์—ฌ ์ƒํƒœ๋ฅผ ๊ฐฑ์‹ ํ•œ๋‹ค.
  • ์ƒํƒœ๋Š” State ๊ฐ์ฒด ์•ˆ์—์„œ ์œ ์ง€๋˜๋ฉฐ, ์œ„์ ฏ์ด ์ œ๊ฑฐ๋  ๋•Œ ํ•จ๊ป˜ ์‚ฌ๋ผ์ง„๋‹ค.
  • ์•ฑ์˜ ์ผ๋ถ€๋ถ„์—์„œ๋งŒ ํ•„์š”ํ•œ ์ƒํƒœ(๋‹จ๊ธฐ ์ƒํƒœ, ephemeral state)์— ์ ํ•ฉํ•˜๋‹ค.

์ด๋Ÿฌํ•œ ํŠน์„ฑ ๋•Œ๋ฌธ์— StatefulWidget์€ ๋‹จ์ˆœํ•œ UI ์š”์†Œ์—์„œ ์ƒํƒœ๋ฅผ ๊ด€๋ฆฌํ•  ๋•Œ ์ ์ ˆํ•œ ์„ ํƒ์ด ๋œ๋‹ค. ํ•˜์ง€๋งŒ ์—ฌ๋Ÿฌ ์œ„์ ฏ ๊ฐ„์— ์ƒํƒœ๋ฅผ ๊ณต์œ ํ•ด์•ผ ํ•˜๋Š” ๊ฒฝ์šฐ์—๋Š” ๋‹ค๋ฅธ ๋ฐฉ๋ฒ•์„ ๊ณ ๋ คํ•ด์•ผ ํ•œ๋‹ค.

class CounterWidget extends StatefulWidget {
  const CounterWidget({super.key});

  @override
  State<CounterWidget> createState() => _CounterWidgetState();
}

class _CounterWidgetState extends State<CounterWidget> {
  int count = 0; // ์ƒํƒœ ๋ณ€์ˆ˜ ์„ ์–ธ

  void increment() {
    setState(() { // ์ƒํƒœ ๋ณ€๊ฒฝ ์‹œ UI ์—…๋ฐ์ดํŠธ
      count++;
    });
  }

  @override
  Widget build(BuildContext context) {
    return Column(
      children: [
        Text('Count: $count'),
        TextButton(
          onPressed: increment,
          child: Text('Increment'),
        ),
      ],
    );
  }
}

 

 

  1. StatefulWidget์€ ์œ„์ ฏ์ด ์ƒํƒœ๋ฅผ ๋ณด๊ด€ํ•  ์ˆ˜ ์žˆ๋„๋ก State ๊ฐ์ฒด๋ฅผ ์ƒ์„ฑํ•œ๋‹ค.
  2. setState()๋ฅผ ํ˜ธ์ถœํ•˜๋ฉด Flutter๊ฐ€ build()๋ฅผ ๋‹ค์‹œ ์‹คํ–‰ํ•˜์—ฌ UI๋ฅผ ์—…๋ฐ์ดํŠธํ•œ๋‹ค.
  3. ์ƒํƒœ๋Š” ์œ„์ ฏ์ด ํ™”๋ฉด์—์„œ ์‚ฌ๋ผ์ง€๋ฉด ํ•จ๊ป˜ ์ œ๊ฑฐ๋œ๋‹ค.

๐Ÿ“Œ ์–ธ์ œ ์‚ฌ์šฉํ• ๊นŒ?

  • ํŠน์ • ์œ„์ ฏ์—์„œ๋งŒ ํ•„์š”ํ•œ ์ƒํƒœ
  • ์™ธ๋ถ€ ์œ„์ ฏ๊ณผ ๊ณต์œ ํ•  ํ•„์š”๊ฐ€ ์—†๋Š” ๋ฐ์ดํ„ฐ

 

3. ์œ„์ ฏ ๊ฐ„ ์ƒํƒœ ๊ณต์œ  ๋ฐฉ๋ฒ•

์œ„์ ฏ ๊ฐ„ ์ƒํƒœ๋ฅผ ๊ณต์œ ํ•ด์•ผ ํ•  ๊ฒฝ์šฐ, StatefulWidget๋งŒ์œผ๋กœ๋Š” ํ•œ๊ณ„๊ฐ€ ์žˆ๋‹ค. Flutter์—์„œ๋Š” ์œ„์ ฏ ํŠธ๋ฆฌ๋ฅผ ํ†ตํ•ด ๋ฐ์ดํ„ฐ๋ฅผ ์ „๋‹ฌํ•˜๋Š” ๋ฐฉ๋ฒ•์„ ์ œ๊ณตํ•œ๋‹ค.

3.1 ์ƒ์„ฑ์ž(Constructor)๋กœ ์ƒํƒœ ์ „๋‹ฌ

์œ„์ ฏ ๊ฐ„ ๋ฐ์ดํ„ฐ๋ฅผ ์ „๋‹ฌํ•˜๋Š” ๊ฐ€์žฅ ๊ฐ„๋‹จํ•œ ๋ฐฉ๋ฒ•์€ ์ƒ์„ฑ์ž๋ฅผ ํ†ตํ•ด ์ „๋‹ฌํ•˜๋Š” ๊ฒƒ์ด๋‹ค.

import 'package:flutter/material.dart';

void main() {
  runApp(const MyApp());
}

class MyApp extends StatelessWidget {
  const MyApp({super.key});

  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      home: ParentWidget(),
    );
  }
}
class CounterWidget extends StatefulWidget {
  const CounterWidget({super.key});

  @override
  State<CounterWidget> createState() => _CounterWidgetState();
}

class _CounterWidgetState extends State<CounterWidget> {
  int count = 0; // ์ƒํƒœ ๋ณ€์ˆ˜ ์„ ์–ธ

  void increment() {
    setState(() { // ์ƒํƒœ ๋ณ€๊ฒฝ ์‹œ UI ์—…๋ฐ์ดํŠธ
      count++;
    });
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: const Text("StatefulWidget ์ƒํƒœ ๊ด€๋ฆฌ")),
      body: Center(
        child: Column(
          mainAxisAlignment: MainAxisAlignment.center,
          children: [
            Text('Count: $count', style: TextStyle(fontSize: 24)),
            const SizedBox(height: 20),
            ElevatedButton(
              onPressed: increment,
              child: const Text('Increment'),
            ),
          ],
        ),
      ),
    );
  }
}

๐Ÿ“Œ ์–ธ์ œ ์‚ฌ์šฉํ• ๊นŒ?

  • ๋ถ€๋ชจ์—์„œ ์ž์‹ ์œ„์ ฏ์œผ๋กœ ๋ฐ์ดํ„ฐ๋ฅผ ์ „๋‹ฌํ•  ๋•Œ
  • ํ•œ๋‘ ๊ฐœ์˜ ์œ„์ ฏ๋งŒ ์ƒํƒœ๋ฅผ ๊ณต์œ ํ•  ๊ฒฝ์šฐ

 

3.2 InheritedWidget์„ ํ™œ์šฉํ•œ ์ƒํƒœ ๊ณต์œ 

InheritedWidget์„ ์‚ฌ์šฉํ•˜๋ฉด ๋ถ€๋ชจ ์œ„์ ฏ์—์„œ ํ•˜์œ„ ์œ„์ ฏ์œผ๋กœ ๋ฐ์ดํ„ฐ๋ฅผ ์ „๋‹ฌํ•  ๋•Œ, ์ง์ ‘์ ์ธ ์ƒ์„ฑ์ž ์ „๋‹ฌ ์—†์ด๋„ ์ƒํƒœ๋ฅผ ๊ณต์œ ํ•  ์ˆ˜ ์žˆ๋‹ค.

import 'package:flutter/material.dart';

void main() {
  runApp(const MyApp()); // ์•ฑ ์‹คํ–‰
}

// ์ตœ์ƒ์œ„ ์œ„์ ฏ
class MyApp extends StatelessWidget {
  const MyApp({super.key});

  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      home: CounterState( // CounterState๋ฅผ ๊ฐ์‹ธ์„œ ์ƒํƒœ๋ฅผ ๊ณต์œ ํ•  ์ˆ˜ ์žˆ๋„๋ก ์„ค์ •
        count: 0, // ์ดˆ๊ธฐ count ๊ฐ’ 0
        child: const CounterScreen(), // ํ•˜์œ„ ์œ„์ ฏ์œผ๋กœ ์ „๋‹ฌ
      ),
    );
  }
// ์ƒํƒœ๋ฅผ ์ €์žฅํ•˜๋Š” InheritedWidget
class CounterState extends InheritedWidget {
  final int count; // ๊ณต์œ ํ•  ์ƒํƒœ ๋ณ€์ˆ˜

  // ์ƒ์„ฑ์ž์—์„œ count ๊ฐ’์„ ์ „๋‹ฌ๋ฐ›์•„ ์ €์žฅ
  const CounterState({super.key, required this.count, required super.child});

  // ํ•˜์œ„ ์œ„์ ฏ์ด CounterState์— ์ ‘๊ทผํ•˜๋Š” ๋ฉ”์„œ๋“œ
  static CounterState of(BuildContext context) {
    return context.dependOnInheritedWidgetOfExactType<CounterState>()!;
  }

  // count ๊ฐ’์ด ๋ณ€๊ฒฝ๋˜์—ˆ์„ ๋•Œ UI๋ฅผ ๋‹ค์‹œ ๋นŒ๋“œํ• ์ง€ ๊ฒฐ์ •ํ•˜๋Š” ๋ฉ”์„œ๋“œ
  @override
  bool updateShouldNotify(CounterState oldWidget) => count != oldWidget.count;
}
// UI๋ฅผ ํ‘œ์‹œํ•˜๋Š” CounterScreen
class CounterScreen extends StatelessWidget {
  const CounterScreen({super.key});

  @override
  Widget build(BuildContext context) {
    int count = CounterState.of(context).count; // InheritedWidget์—์„œ count ๊ฐ’ ๊ฐ€์ ธ์˜ค๊ธฐ

    return Scaffold(
      appBar: AppBar(title: const Text("InheritedWidget ์ƒํƒœ ๊ณต์œ ")),
      body: Center(
        child: Text(
          'Count: $count', // count ๊ฐ’์„ UI์— ํ‘œ์‹œ
          style: TextStyle(fontSize: 24),
        ),
      ),
    );
  }
}

๐Ÿ“Œ ์–ธ์ œ ์‚ฌ์šฉํ• ๊นŒ?

  • ์—ฌ๋Ÿฌ ์œ„์ ฏ์—์„œ ๋™์ผํ•œ ์ƒํƒœ๋ฅผ ์‚ฌ์šฉํ•ด์•ผ ํ•  ๋•Œ
  • ๋ถ€๋ชจ ์œ„์ ฏ์„ ๊ฑฐ์น˜์ง€ ์•Š๊ณ  ํ•˜์œ„ ์œ„์ ฏ์—์„œ ๋ฐ์ดํ„ฐ๋ฅผ ๊ฐ€์ ธ์™€์•ผ ํ•  ๋•Œ
  • ์„ฑ๋Šฅ ์ตœ์ ํ™”๋ฅผ ์œ„ํ•ด ํ•„์š”ํ•  ๋•Œ (์ž์ฃผ ๋ณ€๊ฒฝ๋˜์ง€ ์•Š๋Š” ์ƒํƒœ์— ์ ํ•ฉ)

 

4. Listenable์„ ํ™œ์šฉํ•œ ์ƒํƒœ ๊ด€๋ฆฌ

Flutter์—์„œ๋Š” UI๊ฐ€ ์ƒํƒœ(state) ๋ณ€๊ฒฝ์„ ๊ฐ์ง€ํ•˜๊ณ  ์ž๋™์œผ๋กœ ์—…๋ฐ์ดํŠธ๋˜๋„๋ก Listenable์„ ์‚ฌ์šฉํ•  ์ˆ˜ ์žˆ๋‹ค.
Listenable์€ ์—ฌ๋Ÿฌ ์œ„์ ฏ์ด ๋™์ผํ•œ ์ƒํƒœ๋ฅผ ๊ฐ์ง€ํ•˜๊ณ , ์ƒํƒœ ๋ณ€๊ฒฝ์ด ๋ฐœ์ƒํ•˜๋ฉด UI๋ฅผ ์ž๋™์œผ๋กœ ๊ฐฑ์‹ ํ•  ์ˆ˜ ์žˆ๋„๋ก ํ•œ๋‹ค.

Listenable์€ Flutter์—์„œ ์ƒํƒœ ๋ณ€๊ฒฝ์„ ๊ฐ์ง€ํ•˜๋Š” ๊ธฐ๋ณธ์ ์ธ ์ธํ„ฐํŽ˜์ด์Šค๋‹ค.
์—ฌ๋Ÿฌ ์œ„์ ฏ์ด ๋™์ผํ•œ ์ƒํƒœ๋ฅผ ์ฐธ์กฐํ•˜๊ณ  ์žˆ๋‹ค๊ฐ€ ์ƒํƒœ๊ฐ€ ๋ณ€๊ฒฝ๋˜๋ฉด UI๋ฅผ ์ž๋™์œผ๋กœ ๋‹ค์‹œ ๋นŒ๋“œํ•  ์ˆ˜ ์žˆ๋‹ค.

4.1. ChangeNotifier

ChangeNotifier๋Š” ์ƒํƒœ ๋ณ€๊ฒฝ์ด ๋ฐœ์ƒํ•  ๋•Œ notifyListeners()๋ฅผ ํ˜ธ์ถœํ•˜์—ฌ UI๋ฅผ ์ž๋™์œผ๋กœ ๊ฐฑ์‹ ํ•˜๋„๋ก ํ•˜๋Š” ํด๋ž˜์Šค์ด๋‹ค.

  1. CounterNotifier ํด๋ž˜์Šค๊ฐ€ ChangeNotifier๋ฅผ ์ƒ์†ํ•˜์—ฌ count ์ƒํƒœ๋ฅผ ๊ด€๋ฆฌ.
  2. increment() ํ•จ์ˆ˜์—์„œ count ๊ฐ’์„ ๋ณ€๊ฒฝํ•˜๊ณ  notifyListeners()๋ฅผ ํ˜ธ์ถœํ•˜์—ฌ ๋ณ€๊ฒฝ ์‚ฌํ•ญ์„ UI์— ๋ฐ˜์˜.
  3. ListenableBuilder๋ฅผ ์‚ฌ์šฉํ•˜์—ฌ ์ƒํƒœ ๋ณ€๊ฒฝ ์‹œ build()๊ฐ€ ๋‹ค์‹œ ์‹คํ–‰๋จ.
import 'package:flutter/material.dart';

void main() {
  runApp(const MyApp());
}

// ์•ฑ์˜ ๋ฃจํŠธ ์œ„์ ฏ
class MyApp extends StatelessWidget {
  const MyApp({super.key});

  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      home: ChangeNotifierExample(), // ChangeNotifier ์˜ˆ์ œ ์‹คํ–‰
    );
  }
}
// ChangeNotifier๋ฅผ ์ƒ์†๋ฐ›์•„ ์ƒํƒœ ๊ด€๋ฆฌ
class CounterNotifier extends ChangeNotifier {
  int _count = 0; // ์ƒํƒœ ๋ณ€์ˆ˜

  int get count => _count; // count ๊ฐ’์„ ์™ธ๋ถ€์—์„œ ์ฝ์„ ์ˆ˜ ์žˆ๋„๋ก getter ์ œ๊ณต

  void increment() {
    _count++; // ์ƒํƒœ ๊ฐ’ ๋ณ€๊ฒฝ
    notifyListeners(); // UI ์—…๋ฐ์ดํŠธ๋ฅผ ์œ„ํ•ด ๋ชจ๋“  ๋ฆฌ์Šค๋„ˆ์—๊ฒŒ ๋ณ€๊ฒฝ ์•Œ๋ฆผ
  }
}
class ChangeNotifierExample extends StatefulWidget {
  const ChangeNotifierExample({super.key});

  @override
  _ChangeNotifierExampleState createState() => _ChangeNotifierExampleState();
}

class _ChangeNotifierExampleState extends State<ChangeNotifierExample> {
  final CounterNotifier counterNotifier = CounterNotifier(); // ChangeNotifier ์ธ์Šคํ„ด์Šค ์ƒ์„ฑ

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: const Text("ChangeNotifier ์ƒํƒœ ๊ด€๋ฆฌ")),
      body: Column(
        mainAxisAlignment: MainAxisAlignment.center,
        children: [
          // ListenableBuilder๋ฅผ ์‚ฌ์šฉํ•˜์—ฌ ์ƒํƒœ ๋ณ€๊ฒฝ ๊ฐ์ง€
          ListenableBuilder(
            listenable: counterNotifier, // listenable์— counterNotifier ๋“ฑ๋ก
            builder: (context, child) {
              return Text(
                'Counter: ${counterNotifier.count}', // ์ƒํƒœ ๊ฐ’ ํ‘œ์‹œ
                style: TextStyle(fontSize: 24),
              );
            },
          ),
          const SizedBox(height: 20),
          ElevatedButton(
            onPressed: counterNotifier.increment, // ๋ฒ„ํŠผ ํด๋ฆญ ์‹œ ์ƒํƒœ ๋ณ€๊ฒฝ
            child: const Text('Increment'),
          ),
        ],
      ),
    );
  }
}

 

4.2. ValueNotifier

ValueNotifier๋Š” ChangeNotifier๋ณด๋‹ค ๋” ๊ฐ€๋ฒผ์šด ์ƒํƒœ ๊ด€๋ฆฌ ๋ฐฉ๋ฒ•์ด๋‹ค.

๋‹จ ํ•˜๋‚˜์˜ ๊ฐ’๋งŒ ๊ด€๋ฆฌํ•  ๋•Œ ์ ํ•ฉํ•˜๋ฉฐ, notifyListeners() ์—†์ด๋„ UI๊ฐ€ ์ž๋™์œผ๋กœ ์—…๋ฐ์ดํŠธ๋œ๋‹ค.

  1. ValueNotifier<int> ํƒ€์ž…์˜ counterNotifier ๊ฐ์ฒด ์ƒ์„ฑ.
  2. counterNotifier.value++์„ ์‚ฌ์šฉํ•˜์—ฌ ์ƒํƒœ ๊ฐ’์„ ๋ณ€๊ฒฝํ•˜๋ฉด ์ž๋™์œผ๋กœ UI๊ฐ€ ์—…๋ฐ์ดํŠธ๋จ.
  3. ValueListenableBuilder๋ฅผ ์‚ฌ์šฉํ•˜์—ฌ ์ƒํƒœ๊ฐ€ ๋ณ€๊ฒฝ๋  ๋•Œ๋งŒ ๋‹ค์‹œ ๋นŒ๋“œ.
import 'package:flutter/material.dart';

void main() {
  runApp(const MyApp());
}

// ์•ฑ์˜ ๋ฃจํŠธ ์œ„์ ฏ
class MyApp extends StatelessWidget {
  const MyApp({super.key});

  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      home: ValueNotifierExample(), // ValueNotifier ์˜ˆ์ œ ์‹คํ–‰
    );
  }
}
class ValueNotifierExample extends StatefulWidget {
  const ValueNotifierExample({super.key});

  @override
  _ValueNotifierExampleState createState() => _ValueNotifierExampleState();
}

class _ValueNotifierExampleState extends State<ValueNotifierExample> {
  final ValueNotifier<int> counterNotifier = ValueNotifier(0); // ValueNotifier ์„ ์–ธ

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: const Text("ValueNotifier ์ƒํƒœ ๊ด€๋ฆฌ")),
      body: Column(
        mainAxisAlignment: MainAxisAlignment.center,
        children: [
          // ValueListenableBuilder๋ฅผ ์‚ฌ์šฉํ•˜์—ฌ ์ƒํƒœ ๋ณ€๊ฒฝ ๊ฐ์ง€
          ValueListenableBuilder<int>(
            valueListenable: counterNotifier, // counterNotifier์˜ ๋ณ€๊ฒฝ์„ ๊ฐ์ง€
            builder: (context, value, child) {
              return Text(
                'Counter: $value', // ํ˜„์žฌ ์ƒํƒœ ๊ฐ’ ํ‘œ์‹œ
                style: TextStyle(fontSize: 24),
              );
            },
          ),
          const SizedBox(height: 20),
          ElevatedButton(
            onPressed: () {
              counterNotifier.value++; // ์ƒํƒœ ๊ฐ’์„ ๋ณ€๊ฒฝํ•˜๋ฉด ์ž๋™์œผ๋กœ UI ์—…๋ฐ์ดํŠธ
            },
            child: const Text('Increment'),
          ),
        ],
      ),
    );
  }
}

 

5. MVVM ์•„ํ‚คํ…์ฒ˜ ์ ์šฉ

MVVM(Model-View-ViewModel) ํŒจํ„ด์ด๋ž€?

MVVM ํŒจํ„ด์€ Flutter์™€ ๊ฐ™์€ ๋ฆฌ์•กํ‹ฐ๋ธŒ ํ”„๋ ˆ์ž„์›Œํฌ์—์„œ ํšจ๊ณผ์ ์œผ๋กœ ์ƒํƒœ๋ฅผ ๊ด€๋ฆฌํ•˜๋Š” ์•„ํ‚คํ…์ฒ˜ ํŒจํ„ด์ด๋‹ค. ์ด ํŒจํ„ด์€ ์•ฑ์˜ UI(View)์™€ ๋ฐ์ดํ„ฐ(Model)๋ฅผ ์ง์ ‘ ์—ฐ๊ฒฐํ•˜์ง€ ์•Š๊ณ , ViewModel์ด ์ค‘๊ฐ„ ์—ญํ• ์„ ํ•˜๋„๋ก ๋ถ„๋ฆฌํ•˜์—ฌ ์ฝ”๋“œ์˜ ์œ ์ง€๋ณด์ˆ˜์„ฑ์„ ๋†’์ธ๋‹ค.

MVVM์˜ ๊ตฌ์„ฑ ์š”์†Œ

  1. Model (๋ชจ๋ธ)
    • ์•ฑ์˜ ๋ฐ์ดํ„ฐ์™€ ๋กœ์ง์„ ๋‹ด๋‹นํ•˜๋Š” ๊ณ„์ธต.
    • HTTP ์š”์ฒญ, ๋ฐ์ดํ„ฐ๋ฒ ์ด์Šค ์ ‘๊ทผ, ๋กœ์ปฌ ์ €์žฅ์†Œ ๋“ฑ์˜ ์ž‘์—…์„ ์ˆ˜ํ–‰.
    • Flutter์˜ UI ๊ด€๋ จ ์ฝ”๋“œ๊ฐ€ ํฌํ•จ๋˜์ง€ ์•Š์Œ → dart:ui์™€ ๋…๋ฆฝ์ ์ž„.
  2. ViewModel (๋ทฐ๋ชจ๋ธ)
    • Model๊ณผ View ์‚ฌ์ด์—์„œ ๋ฐ์ดํ„ฐ ํ๋ฆ„์„ ๊ด€๋ฆฌํ•˜๋Š” ๊ณ„์ธต.
    • ChangeNotifier๋ฅผ ์ƒ์†ํ•˜์—ฌ ์ƒํƒœ๊ฐ€ ๋ณ€๊ฒฝ๋  ๋•Œ UI์— ์•Œ๋ฆผ.
    • ๋ฐ์ดํ„ฐ ๋ณ€ํ™˜, ๋กœ๋”ฉ ์ƒํƒœ ๊ด€๋ฆฌ, ์—๋Ÿฌ ํ•ธ๋“ค๋ง ๋“ฑ์˜ ์—ญํ•  ์ˆ˜ํ–‰.
  3. View (๋ทฐ)
    • UI ์š”์†Œ๋ฅผ ๋‹ด๋‹นํ•˜๋Š” ๊ณ„์ธต.
    • ViewModel์˜ ๋ฐ์ดํ„ฐ๋ฅผ ๊ตฌ๋…ํ•˜์—ฌ UI๋ฅผ ์—…๋ฐ์ดํŠธํ•จ.
    • UI์™€ ๋กœ์ง์„ ๋ถ„๋ฆฌํ•˜์—ฌ ์œ ์ง€๋ณด์ˆ˜์„ฑ์„ ๋†’์ž„.

3.1. Model ์ •์˜

Model์€ ์•ฑ์˜ ๋ฐ์ดํ„ฐ ๋กœ์ง์„ ์ฒ˜๋ฆฌํ•˜๋Š” ์—ญํ• ์„ ํ•œ๋‹ค. ์˜ˆ์ œ์—์„œ๋Š” CounterModel์ด ์„œ๋ฒ„์—์„œ ๋ฐ์ดํ„ฐ๋ฅผ ๊ฐ€์ ธ์˜ค๊ณ  ์—…๋ฐ์ดํŠธํ•˜๋Š” ์—ญํ• ์„ ์ˆ˜ํ–‰ํ•œ๋‹ค.

import 'package:http/http.dart';

class CounterData {
  CounterData(this.count);
  final int count;
}

class CounterModel {
  Future<CounterData> loadCountFromServer() async {
    final uri = Uri.parse('https://myfluttercounterapp.net/count');
    final response = await get(uri);

    if (response.statusCode != 200) {
      throw ('Failed to fetch data');
    }
    return CounterData(int.parse(response.body));
  }

  Future<CounterData> updateCountOnServer(int newCount) async {
    // ์„œ๋ฒ„๋กœ newCount ์ „์†กํ•˜๋Š” ์ฝ”๋“œ ์ž‘์„ฑ ํ•„์š”
  }
}

โœ” Model์˜ ํŠน์ง•

  • HTTP ์š”์ฒญ์„ ํ†ตํ•ด ์„œ๋ฒ„์—์„œ ๋ฐ์ดํ„ฐ๋ฅผ ๊ฐ€์ ธ์˜ค๊ฑฐ๋‚˜ ๋ณ€๊ฒฝํ•จ.
  • Flutter ํ”„๋ ˆ์ž„์›Œํฌ์™€ ๋…๋ฆฝ์ ์œผ๋กœ ๋™์ž‘ํ•จ (ํ…Œ์ŠคํŠธ ์šฉ์ด).
  • ViewModel์ด ์ง์ ‘ ๋ฐ์ดํ„ฐ๋ฅผ ์ ‘๊ทผํ•˜์ง€ ์•Š๊ณ  Model์„ ํ†ตํ•ด ๊ด€๋ฆฌํ•จ.

 

3.2. ViewModel ์ •์˜

ViewModel์€ Model์˜ ๋ฐ์ดํ„ฐ๋ฅผ ๊ด€๋ฆฌํ•˜๊ณ  UI(View)์—์„œ ์‚ฌ์šฉํ•  ์ˆ˜ ์žˆ๋„๋ก ์ œ๊ณตํ•˜๋Š” ์—ญํ• ์„ ํ•œ๋‹ค.

import 'package:flutter/foundation.dart';

class CounterViewModel extends ChangeNotifier {
  final CounterModel model;
  int? count;
  String? errorMessage;

  CounterViewModel(this.model);

  Future<void> init() async {
    try {
      count = (await model.loadCountFromServer()).count;
    } catch (e) {
      errorMessage = 'Failed to load data';
    }
    notifyListeners();
  }

  Future<void> increment() async {
    var currentCount = count;
    if (currentCount == null) {
      throw ('Not initialized');
    }
    try {
      await model.updateCountOnServer(currentCount + 1);
      count++;
    } catch (e) {
      errorMessage = 'Failed to update count';
    }
    notifyListeners();
  }
}

โœ” ViewModel์˜ ํŠน์ง•

  • ChangeNotifier๋ฅผ ์ƒ์†ํ•˜์—ฌ ์ƒํƒœ ๋ณ€๊ฒฝ ์‹œ UI๋ฅผ ์—…๋ฐ์ดํŠธํ•จ.
  • Model๊ณผ View๋ฅผ ๋ถ„๋ฆฌํ•˜์—ฌ UI ๋กœ์ง๊ณผ ๋ฐ์ดํ„ฐ ์ฒ˜๋ฆฌ๋ฅผ ๋ถ„๋ฆฌํ•จ.
  • ์„œ๋ฒ„์—์„œ ๋ฐ์ดํ„ฐ๋ฅผ ๊ฐ€์ ธ์˜ค๊ณ , ์ƒํƒœ๋ฅผ ๋ณ€๊ฒฝํ•˜๋Š” ์—ญํ• ์„ ์ˆ˜ํ–‰ํ•จ.

 

3.3. View ์ •์˜

View๋Š” UI๋ฅผ ๋‹ด๋‹นํ•˜๋ฉฐ, ViewModel์˜ ๋ฐ์ดํ„ฐ๋ฅผ ๊ตฌ๋…ํ•˜์—ฌ ์ž๋™์œผ๋กœ UI๋ฅผ ์—…๋ฐ์ดํŠธํ•œ๋‹ค.

ListenableBuilder(
  listenable: viewModel,
  builder: (context, child) {
    return Column(
      children: [
        if (viewModel.errorMessage != null)
          Text(
            'Error: ${viewModel.errorMessage}',
            style: TextStyle(color: Colors.red),
          ),
        Text('Count: ${viewModel.count}'),
        TextButton(
          onPressed: viewModel.increment,
          child: Text('Increment'),
        ),
      ],
    );
  },
)

โœ” View์˜ ํŠน์ง•

  • ViewModel์˜ ๋ฐ์ดํ„ฐ๋ฅผ ๊ตฌ๋…ํ•˜์—ฌ UI๋ฅผ ์—…๋ฐ์ดํŠธํ•จ.
  • ListenableBuilder๋ฅผ ์‚ฌ์šฉํ•˜์—ฌ ์ƒํƒœ๊ฐ€ ๋ณ€๊ฒฝ๋  ๋•Œ UI๋ฅผ ์ž๋™์œผ๋กœ ๊ฐฑ์‹ ํ•จ.
  • UI ๋กœ์ง์„ ์ตœ์†Œํ™”ํ•˜์—ฌ ์œ ์ง€๋ณด์ˆ˜์„ฑ์„ ๋†’์ž„.

MVVM ํŒจํ„ด์„ ์‚ฌ์šฉํ•˜๋ฉด UI(View)์™€ ๋ฐ์ดํ„ฐ(Model)๋ฅผ ๋ถ„๋ฆฌํ•˜์—ฌ ์ฝ”๋“œ์˜ ์œ ์ง€๋ณด์ˆ˜์„ฑ๊ณผ ํ™•์žฅ์„ฑ์„ ๋†’์ผ ์ˆ˜ ์žˆ๋‹ค.
Flutter์—์„œ ์ƒํƒœ ๊ด€๋ฆฌ๋ฅผ ์ฒด๊ณ„์ ์œผ๋กœ ํ•˜๊ณ  ์‹ถ๋‹ค๋ฉด MVVM ํŒจํ„ด์„ ๋„์ž…ํ•˜๋Š” ๊ฒƒ์ด ์ข‹์€ ์„ ํƒ์ด ๋  ์ˆ˜ ์žˆ๋‹ค.

๋ฐ˜์‘ํ˜•