์ผ | ์ | ํ | ์ | ๋ชฉ | ๊ธ | ํ |
---|---|---|---|---|---|---|
1 | 2 | 3 | 4 | 5 | ||
6 | 7 | 8 | 9 | 10 | 11 | 12 |
13 | 14 | 15 | 16 | 17 | 18 | 19 |
20 | 21 | 22 | 23 | 24 | 25 | 26 |
27 | 28 | 29 | 30 |
- ์๋๋ก์ด๋ ์ฑ ๊ฐ๋ฐ
- ํ๋ค์ค๊ณต๋ถ
- Android
- ํ์ด์ฌ
- RESTful API
- ์๋ฐ๊ณต๋ถ
- REACT
- db
- Streamlit๊ธฐ์ด
- Flutter
- ์๋ฐ๊ธฐ์ด
- ์๋ฐ
- java
- serverless
- ํ์ด์ฌ๊ณต๋ถ
- streamlit
- ์คํธ๋ฆผ๋ฆฟ ๊ธฐ๋ณธ
- ๋ฐ์ดํฐ๋ฒ ์ด์ค
- ์๋๋ก์ด๋ ์คํ๋์ค
- ์๋ฐ์คํฌ๋ฆฝํธ
- ์๋ฐ์คํฌ๋ฆฝํธ ๊ณต๋ถ
- JavaScript
- ์น๋์๋ณด๋ ๊ธฐ์ด
- Pandas
- ํ๋ค์ค
- ์คํธ๋ฆผ๋ฆฟ
- ์๋ฐ์ด๋ณด
- MySQL
- ์๋ฐํ๋ก๊ทธ๋๋ฐ
- ์น๋์๋ณด๋ ์ ์
- Today
- Total
ruriruriya
[Flutter] ๊ธฐ๋ณธ ์ํ ๊ด๋ฆฌ (State Management) ๋ณธ๋ฌธ
[Flutter] ๊ธฐ๋ณธ ์ํ ๊ด๋ฆฌ (State Management)
๋ฃจ๋ฆฌ์ผใ 2025. 2. 22. 21:52Flutter์์ ์ํ ๊ด๋ฆฌ๋ UI๊ฐ ๋ณํํ๋ ๋ฐ์ดํฐ๋ฅผ ํจ๊ณผ์ ์ผ๋ก ์ ์งํ๊ณ ๋ฐ์ํ๋ ๋ฐฉ๋ฒ์ด๋ค. ์ํ ๊ด๋ฆฌ๊ฐ ์ค์ํ ์ด์ ๋ Flutter์ ์ ์ธ์ UI ๋ฐฉ์ ๋๋ฌธ์ด๋ค. ์ฆ, UI๋ ์ํ(state)์ ๋ฐ๋ผ ๋ณ๊ฒฝ๋๋ฏ๋ก ์ด๋ฅผ ์ด๋ป๊ฒ ๋ค๋ฃจ๋๋์ ๋ฐ๋ผ ์ฑ์ ๊ตฌ์กฐ์ ์ ์ง๋ณด์์ฑ์ด ๋ฌ๋ผ์ง๋ค.
์ฌ๊ธฐ์๋ Flutter์ ๊ธฐ๋ณธ ์ํ ๊ด๋ฆฌ ๋ฐฉ๋ฒ์ ๋ํด ์ค๋ช ํ๋ค. ์๋ํํฐ ๋ผ์ด๋ธ๋ฌ๋ฆฌ(Provider, Riverpod ๋ฑ)๋ฅผ ์ฌ์ฉํ์ง ์๊ณ Flutter์์ ์ ๊ณตํ๋ ๋ฐฉ๋ฒ๋ง ๋ค๋ฃฌ๋ค.
1. ์ํ(State)๋?
Flutter์์ ์ํ๋ ์ฑ์ UI๋ฅผ ๊ฒฐ์ ํ๋ ๋ฐ์ดํฐ์ด๋ค. ์ํ๋ ํฌ๊ฒ ๋ ๊ฐ์ง๋ก ๋๋๋ค:
- Ephemeral state (๋จ๊ธฐ ์ํ): ํน์ ์์ ฏ ๋ด์์๋ง ๊ด๋ฆฌ๋๋ ์ํ (์: TextField ์ ๋ ฅ๊ฐ, ๋ฒํผ ํด๋ฆญ ์ฌ๋ถ ๋ฑ)
- 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'),
),
],
);
}
}
- StatefulWidget์ ์์ ฏ์ด ์ํ๋ฅผ ๋ณด๊ดํ ์ ์๋๋ก State ๊ฐ์ฒด๋ฅผ ์์ฑํ๋ค.
- setState()๋ฅผ ํธ์ถํ๋ฉด Flutter๊ฐ build()๋ฅผ ๋ค์ ์คํํ์ฌ UI๋ฅผ ์ ๋ฐ์ดํธํ๋ค.
- ์ํ๋ ์์ ฏ์ด ํ๋ฉด์์ ์ฌ๋ผ์ง๋ฉด ํจ๊ป ์ ๊ฑฐ๋๋ค.
๐ ์ธ์ ์ฌ์ฉํ ๊น?
- ํน์ ์์ ฏ์์๋ง ํ์ํ ์ํ
- ์ธ๋ถ ์์ ฏ๊ณผ ๊ณต์ ํ ํ์๊ฐ ์๋ ๋ฐ์ดํฐ
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๋ฅผ ์๋์ผ๋ก ๊ฐฑ์ ํ๋๋ก ํ๋ ํด๋์ค์ด๋ค.
- CounterNotifier ํด๋์ค๊ฐ ChangeNotifier๋ฅผ ์์ํ์ฌ count ์ํ๋ฅผ ๊ด๋ฆฌ.
- increment() ํจ์์์ count ๊ฐ์ ๋ณ๊ฒฝํ๊ณ notifyListeners()๋ฅผ ํธ์ถํ์ฌ ๋ณ๊ฒฝ ์ฌํญ์ UI์ ๋ฐ์.
- 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๊ฐ ์๋์ผ๋ก ์ ๋ฐ์ดํธ๋๋ค.
- ValueNotifier<int> ํ์ ์ counterNotifier ๊ฐ์ฒด ์์ฑ.
- counterNotifier.value++์ ์ฌ์ฉํ์ฌ ์ํ ๊ฐ์ ๋ณ๊ฒฝํ๋ฉด ์๋์ผ๋ก UI๊ฐ ์ ๋ฐ์ดํธ๋จ.
- 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์ ๊ตฌ์ฑ ์์
- Model (๋ชจ๋ธ)
- ์ฑ์ ๋ฐ์ดํฐ์ ๋ก์ง์ ๋ด๋นํ๋ ๊ณ์ธต.
- HTTP ์์ฒญ, ๋ฐ์ดํฐ๋ฒ ์ด์ค ์ ๊ทผ, ๋ก์ปฌ ์ ์ฅ์ ๋ฑ์ ์์ ์ ์ํ.
- Flutter์ UI ๊ด๋ จ ์ฝ๋๊ฐ ํฌํจ๋์ง ์์ → dart:ui์ ๋ ๋ฆฝ์ ์.
- ViewModel (๋ทฐ๋ชจ๋ธ)
- Model๊ณผ View ์ฌ์ด์์ ๋ฐ์ดํฐ ํ๋ฆ์ ๊ด๋ฆฌํ๋ ๊ณ์ธต.
- ChangeNotifier๋ฅผ ์์ํ์ฌ ์ํ๊ฐ ๋ณ๊ฒฝ๋ ๋ UI์ ์๋ฆผ.
- ๋ฐ์ดํฐ ๋ณํ, ๋ก๋ฉ ์ํ ๊ด๋ฆฌ, ์๋ฌ ํธ๋ค๋ง ๋ฑ์ ์ญํ ์ํ.
- 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 ํจํด์ ๋์
ํ๋ ๊ฒ์ด ์ข์ ์ ํ์ด ๋ ์ ์๋ค.
'๐ฑFlutter > Flutter Framework' ์นดํ ๊ณ ๋ฆฌ์ ๋ค๋ฅธ ๊ธ
[Flutter] ์ํ ๊ด๋ฆฌ ๋ผ์ด๋ธ๋ฌ๋ฆฌ Provider ํจํค์ง (0) | 2025.02.25 |
---|---|
[Flutter] Flutter ๋ ์ด์์ ์ดํดํ๊ธฐ(Layout) (0) | 2025.02.18 |
[Flutter] ๊ธฐ๋ณธ ๋ ์ด์์์ ๊ตฌ์ฑํ๋ Scaffold ์์ ฏ (0) | 2025.01.20 |
[Flutter] build ๋ฉ์๋ (0) | 2025.01.17 |
[Flutter] ์์ ฏ์ ๋ํด์(Widget) (0) | 2025.01.15 |