@k2works/claude-code-booster 0.1.3 → 0.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +14 -0
- package/bin/claude-code-booster +24 -4
- package/lib/assets/.claude/README.md +44 -40
- package/lib/assets/.claude/commands/analysis.md +230 -0
- package/lib/assets/.claude/commands/kill.md +109 -0
- package/lib/assets/.claude/commands/next.md +136 -0
- package/lib/assets/.claude/commands/plan.md +141 -91
- package/lib/assets/.claude/commands/progress.md +172 -0
- package/lib/assets/docs/reference/UI/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +446 -0
- package/lib/assets/docs/reference//343/202/242/343/203/274/343/202/255/343/203/206/343/202/257/343/203/201/343/203/243/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +1428 -0
- package/lib/assets/docs/reference//343/202/244/343/203/263/343/203/225/343/203/251/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +1879 -0
- package/lib/assets/docs/reference//343/203/206/343/202/271/343/203/210/346/210/246/347/225/245/343/202/254/343/202/244/343/203/211.md +1310 -0
- package/lib/assets/docs/reference//343/203/207/343/203/274/343/202/277/343/203/242/343/203/207/343/203/253/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +312 -0
- package/lib/assets/docs/reference//343/203/211/343/203/241/343/202/244/343/203/263/343/203/242/343/203/207/343/203/253/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +600 -0
- package/lib/assets/docs/reference//343/203/246/343/203/274/343/202/271/343/202/261/343/203/274/343/202/271/344/275/234/346/210/220/343/202/254/343/202/244/343/203/211.md +672 -0
- package/lib/assets/docs/reference//343/203/252/343/203/252/343/203/274/343/202/271/343/203/273/343/202/244/343/203/206/343/203/254/343/203/274/343/202/267/343/203/247/343/203/263/350/250/210/347/224/273/343/202/254/343/202/244/343/203/211.md +524 -0
- package/lib/assets/docs/reference//351/201/213/347/224/250/350/246/201/344/273/266/345/256/232/347/276/251/343/202/254/343/202/244/343/203/211.md +393 -0
- package/lib/assets/docs/reference//351/226/213/347/231/272/343/202/254/343/202/244/343/203/211.md +18 -173
- package/lib/assets/docs/reference//351/235/236/346/251/237/350/203/275/350/246/201/344/273/266/345/256/232/347/276/251/343/202/254/343/202/244/343/203/211.md +1231 -0
- package/lib/assets/docs/template//345/256/214/345/205/250/345/275/242/345/274/217/343/201/256/343/203/246/343/203/274/343/202/271/343/202/261/343/203/274/343/202/271.md +64 -0
- package/lib/assets/docs/template//350/246/201/344/273/266/345/256/232/347/276/251.md +467 -443
- package/package.json +1 -1
|
@@ -0,0 +1,1428 @@
|
|
|
1
|
+
# アーキテクチャ設計ガイド
|
|
2
|
+
|
|
3
|
+
## 概要
|
|
4
|
+
|
|
5
|
+
本ガイドは、システム開発におけるアーキテクチャ設計の指針を提供します。バックエンドでは業務領域の特性とデータ構造の複雑さに基づいたアーキテクチャパターンの選択を、フロントエンドでは現代的なWebアプリケーション開発のベストプラクティスを定義します。
|
|
6
|
+
|
|
7
|
+
## アーキテクチャ設計の原則
|
|
8
|
+
|
|
9
|
+
### 基本方針
|
|
10
|
+
|
|
11
|
+
- **モノレポ構成**: 単一リポジトリでフロントエンド・バックエンドを統合管理
|
|
12
|
+
- **関心事の分離**: 明確な責務分離によるメンテナンス性向上
|
|
13
|
+
- **依存関係の方向性**: 外側から内側(ドメイン)への単方向依存
|
|
14
|
+
- **テストしやすい設計**: 各層が独立してテスト可能な構造
|
|
15
|
+
|
|
16
|
+
## 全体構成
|
|
17
|
+
|
|
18
|
+
```plantuml
|
|
19
|
+
@startuml
|
|
20
|
+
package "アプリケーション" {
|
|
21
|
+
package "フロントエンド" {
|
|
22
|
+
[Webアプリケーション] as WebApp
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
package "バックエンド" {
|
|
26
|
+
[APIサーバー] as API
|
|
27
|
+
database "データベース" as DB
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
WebApp --> API : HTTP/REST
|
|
31
|
+
API --> DB : SQL/NoSQL
|
|
32
|
+
}
|
|
33
|
+
@endtuml
|
|
34
|
+
```
|
|
35
|
+
|
|
36
|
+
### コンポーネント責務
|
|
37
|
+
|
|
38
|
+
#### フロントエンド
|
|
39
|
+
- **UI/UX**: ユーザーインターフェース・ユーザー体験
|
|
40
|
+
- **状態管理**: アプリケーション状態とサーバー状態の管理
|
|
41
|
+
- **API連携**: バックエンドとのデータ通信
|
|
42
|
+
- **レンダリング**: 効率的な画面描画とユーザーインタラクション
|
|
43
|
+
|
|
44
|
+
#### バックエンド
|
|
45
|
+
- **ビジネスロジック**: 業務ルール・制約の実装
|
|
46
|
+
- **データ永続化**: データの保存・取得・更新・削除
|
|
47
|
+
- **セキュリティ**: 認証・認可・データ保護
|
|
48
|
+
- **外部システム連携**: API・メッセージング・ファイル処理
|
|
49
|
+
|
|
50
|
+
# バックエンドアーキテクチャ
|
|
51
|
+
|
|
52
|
+
## バックエンドアーキテクチャパターン選択フロー
|
|
53
|
+
|
|
54
|
+
```plantuml
|
|
55
|
+
@startuml
|
|
56
|
+
!define DECISION_COLOR #FFE6CC
|
|
57
|
+
!define PROCESS_COLOR #E6F3FF
|
|
58
|
+
!define TERMINAL_COLOR #E6FFE6
|
|
59
|
+
|
|
60
|
+
skinparam roundcorner 10
|
|
61
|
+
skinparam shadowing false
|
|
62
|
+
|
|
63
|
+
start
|
|
64
|
+
|
|
65
|
+
if (業務領域のカテゴリー) then (補完、一般との連携)
|
|
66
|
+
if (データ構造が複雑か?) then (いいえ)
|
|
67
|
+
:トランザクションスクリプト;
|
|
68
|
+
:逆ピラミッド形のテスト;
|
|
69
|
+
if (永続化モデルは複数か?) then (いいえ)
|
|
70
|
+
:レイヤードアーキテクチャ 3層;
|
|
71
|
+
else (はい)
|
|
72
|
+
:レイヤードアーキテクチャ 4層;
|
|
73
|
+
endif
|
|
74
|
+
else (はい)
|
|
75
|
+
:アクティブレコード;
|
|
76
|
+
:ダイヤモンド形のテスト;
|
|
77
|
+
if (永続化モデルは複数か?) then (いいえ)
|
|
78
|
+
:レイヤードアーキテクチャ 3層;
|
|
79
|
+
else (はい)
|
|
80
|
+
:レイヤードアーキテクチャ 4層;
|
|
81
|
+
endif
|
|
82
|
+
endif
|
|
83
|
+
else (中核の業務領域)
|
|
84
|
+
if (金額を扱う/分析/監査記録が必要か?) then (いいえ)
|
|
85
|
+
:ドメインモデル;
|
|
86
|
+
:ピラミッド形のテスト;
|
|
87
|
+
if (永続化モデルは複数か?) then (いいえ)
|
|
88
|
+
:ポートとアダプター;
|
|
89
|
+
else (はい)
|
|
90
|
+
:CQRS;
|
|
91
|
+
endif
|
|
92
|
+
else (はい)
|
|
93
|
+
:イベント履歴式ドメインモデル;
|
|
94
|
+
:ピラミッド形のテスト;
|
|
95
|
+
:CQRS;
|
|
96
|
+
endif
|
|
97
|
+
endif
|
|
98
|
+
|
|
99
|
+
stop
|
|
100
|
+
@enduml
|
|
101
|
+
```
|
|
102
|
+
|
|
103
|
+
### 判断基準
|
|
104
|
+
|
|
105
|
+
#### 業務領域カテゴリー
|
|
106
|
+
|
|
107
|
+
| カテゴリー | 特徴 | 例 |
|
|
108
|
+
|-----------|------|-----|
|
|
109
|
+
| **中核の業務領域** | 組織の競争優位性を決定する領域 | 商品管理、顧客管理、受注処理 |
|
|
110
|
+
| **補完、一般との連携** | 中核を支援する補助的な領域 | レポート出力、バッチ処理、外部連携 |
|
|
111
|
+
|
|
112
|
+
#### データ構造の複雑さ
|
|
113
|
+
|
|
114
|
+
| 複雑さ | 判定基準 |
|
|
115
|
+
|--------|----------|
|
|
116
|
+
| **複雑** | エンティティ間の関係が多く、ビジネスルールが複雑 |
|
|
117
|
+
| **シンプル** | 単純なCRUD操作が中心、関係性が少ない |
|
|
118
|
+
|
|
119
|
+
#### 特殊要件
|
|
120
|
+
|
|
121
|
+
- **金額を扱う**: 会計・決済・金融関連の処理
|
|
122
|
+
- **分析**: 大量データの集計・分析処理
|
|
123
|
+
- **監査記録が必要**: コンプライアンス・規制対応
|
|
124
|
+
|
|
125
|
+
## アーキテクチャパターン詳細
|
|
126
|
+
|
|
127
|
+
### 1. トランザクションスクリプトパターン
|
|
128
|
+
|
|
129
|
+
#### 適用場面
|
|
130
|
+
- **業務領域**: 補完・一般との連携
|
|
131
|
+
- **データ構造**: シンプル
|
|
132
|
+
- **システム規模**: 小規模
|
|
133
|
+
|
|
134
|
+
#### 特徴
|
|
135
|
+
```plantuml
|
|
136
|
+
@startuml
|
|
137
|
+
package "トランザクションスクリプト" {
|
|
138
|
+
[スクリプト1] as s1
|
|
139
|
+
[スクリプト2] as s2
|
|
140
|
+
[スクリプト3] as s3
|
|
141
|
+
database "データベース" as db
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
s1 --> db
|
|
145
|
+
s2 --> db
|
|
146
|
+
s3 --> db
|
|
147
|
+
@enduml
|
|
148
|
+
```
|
|
149
|
+
|
|
150
|
+
**メリット**:
|
|
151
|
+
- 実装が直感的でわかりやすい
|
|
152
|
+
- 開発速度が早い
|
|
153
|
+
- 小規模チームでも扱いやすい
|
|
154
|
+
|
|
155
|
+
**デメリット**:
|
|
156
|
+
- コードの重複が発生しやすい
|
|
157
|
+
- 大規模化すると保守が困難
|
|
158
|
+
- ビジネスロジックが散在する
|
|
159
|
+
|
|
160
|
+
#### 実装指針
|
|
161
|
+
|
|
162
|
+
```java
|
|
163
|
+
// 例:会議室予約処理
|
|
164
|
+
public class ReservationService {
|
|
165
|
+
public void makeReservation(ReservationRequest request) {
|
|
166
|
+
// 1. バリデーション
|
|
167
|
+
validateRequest(request);
|
|
168
|
+
|
|
169
|
+
// 2. 重複チェック
|
|
170
|
+
checkConflicts(request);
|
|
171
|
+
|
|
172
|
+
// 3. 予約作成
|
|
173
|
+
createReservation(request);
|
|
174
|
+
|
|
175
|
+
// 4. 通知送信
|
|
176
|
+
sendNotification(request);
|
|
177
|
+
}
|
|
178
|
+
}
|
|
179
|
+
```
|
|
180
|
+
|
|
181
|
+
### 2. アクティブレコードパターン
|
|
182
|
+
|
|
183
|
+
#### 適用場面
|
|
184
|
+
- **業務領域**: 補完・一般との連携
|
|
185
|
+
- **データ構造**: 複雑
|
|
186
|
+
- **システム規模**: 中規模
|
|
187
|
+
|
|
188
|
+
#### 特徴
|
|
189
|
+
```plantuml
|
|
190
|
+
@startuml
|
|
191
|
+
allow_mixing
|
|
192
|
+
|
|
193
|
+
class User {
|
|
194
|
+
+id: Long
|
|
195
|
+
+name: String
|
|
196
|
+
+email: String
|
|
197
|
+
+save()
|
|
198
|
+
+delete()
|
|
199
|
+
+findById()
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
class Reservation {
|
|
203
|
+
+id: Long
|
|
204
|
+
+userId: Long
|
|
205
|
+
+roomId: Long
|
|
206
|
+
+save()
|
|
207
|
+
+cancel()
|
|
208
|
+
+findByUser()
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
database "データベース" as db
|
|
212
|
+
|
|
213
|
+
User --> db : CRUD操作
|
|
214
|
+
Reservation --> db : CRUD操作
|
|
215
|
+
@enduml
|
|
216
|
+
```
|
|
217
|
+
|
|
218
|
+
**メリット**:
|
|
219
|
+
- オブジェクトとデータベースの対応が明確
|
|
220
|
+
- ORMとの相性が良い
|
|
221
|
+
- 理解しやすい構造
|
|
222
|
+
|
|
223
|
+
**デメリット**:
|
|
224
|
+
- データアクセスとビジネスロジックが密結合
|
|
225
|
+
- テストが困難
|
|
226
|
+
- ドメインロジックが薄くなる傾向
|
|
227
|
+
|
|
228
|
+
#### 実装指針
|
|
229
|
+
|
|
230
|
+
```java
|
|
231
|
+
// 例:予約エンティティ
|
|
232
|
+
@Entity
|
|
233
|
+
public class Reservation {
|
|
234
|
+
@Id
|
|
235
|
+
private Long id;
|
|
236
|
+
private Long userId;
|
|
237
|
+
private Long roomId;
|
|
238
|
+
private LocalDateTime startTime;
|
|
239
|
+
private LocalDateTime endTime;
|
|
240
|
+
|
|
241
|
+
// ビジネスロジックを含む
|
|
242
|
+
public void cancel() {
|
|
243
|
+
if (canCancel()) {
|
|
244
|
+
this.status = CANCELLED;
|
|
245
|
+
this.save();
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
|
|
249
|
+
public boolean canCancel() {
|
|
250
|
+
return startTime.isAfter(LocalDateTime.now().plusHours(2));
|
|
251
|
+
}
|
|
252
|
+
}
|
|
253
|
+
```
|
|
254
|
+
|
|
255
|
+
### 3. ドメインモデルパターン
|
|
256
|
+
|
|
257
|
+
#### 適用場面
|
|
258
|
+
- **業務領域**: 中核の業務領域
|
|
259
|
+
- **特殊要件**: 金額を扱わない、監査記録不要
|
|
260
|
+
- **システム規模**: 中規模〜大規模
|
|
261
|
+
|
|
262
|
+
#### 特徴
|
|
263
|
+
```plantuml
|
|
264
|
+
@startuml
|
|
265
|
+
package "ドメイン層" {
|
|
266
|
+
class User
|
|
267
|
+
class Reservation
|
|
268
|
+
class MeetingRoom
|
|
269
|
+
class ReservationService
|
|
270
|
+
}
|
|
271
|
+
|
|
272
|
+
package "アプリケーション層" {
|
|
273
|
+
class ReservationUseCase
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
package "インフラストラクチャ層" {
|
|
277
|
+
class UserRepository
|
|
278
|
+
class ReservationRepository
|
|
279
|
+
}
|
|
280
|
+
|
|
281
|
+
ReservationUseCase --> ReservationService
|
|
282
|
+
ReservationService --> User
|
|
283
|
+
ReservationService --> Reservation
|
|
284
|
+
ReservationService --> MeetingRoom
|
|
285
|
+
ReservationUseCase --> UserRepository
|
|
286
|
+
ReservationUseCase --> ReservationRepository
|
|
287
|
+
@enduml
|
|
288
|
+
```
|
|
289
|
+
|
|
290
|
+
**メリット**:
|
|
291
|
+
- リッチなドメインモデル
|
|
292
|
+
- ビジネスロジックの集約
|
|
293
|
+
- 高い保守性・拡張性
|
|
294
|
+
- テスト容易性
|
|
295
|
+
|
|
296
|
+
**デメリット**:
|
|
297
|
+
- 初期の学習コストが高い
|
|
298
|
+
- 設計の複雑さ
|
|
299
|
+
- 開発速度が遅い場合がある
|
|
300
|
+
|
|
301
|
+
#### 実装指針
|
|
302
|
+
|
|
303
|
+
```java
|
|
304
|
+
// ドメインモデル例
|
|
305
|
+
public class Reservation {
|
|
306
|
+
private ReservationId id;
|
|
307
|
+
private UserId userId;
|
|
308
|
+
private MeetingRoomId roomId;
|
|
309
|
+
private TimeSlot timeSlot;
|
|
310
|
+
private ReservationStatus status;
|
|
311
|
+
|
|
312
|
+
// ファクトリーメソッド
|
|
313
|
+
public static Reservation create(UserId userId, MeetingRoomId roomId, TimeSlot timeSlot) {
|
|
314
|
+
validateTimeSlot(timeSlot);
|
|
315
|
+
return new Reservation(ReservationId.generate(), userId, roomId, timeSlot, CONFIRMED);
|
|
316
|
+
}
|
|
317
|
+
|
|
318
|
+
// ビジネスロジック
|
|
319
|
+
public void cancel(CancellationPolicy policy) {
|
|
320
|
+
if (!policy.canCancel(this.timeSlot)) {
|
|
321
|
+
throw new CancellationNotAllowedException();
|
|
322
|
+
}
|
|
323
|
+
this.status = CANCELLED;
|
|
324
|
+
}
|
|
325
|
+
}
|
|
326
|
+
```
|
|
327
|
+
|
|
328
|
+
### 4. イベント履歴式ドメインモデルパターン
|
|
329
|
+
|
|
330
|
+
#### 適用場面
|
|
331
|
+
- **業務領域**: 中核の業務領域
|
|
332
|
+
- **特殊要件**: 金額を扱う、分析・監査記録が必要
|
|
333
|
+
- **システム規模**: 大規模
|
|
334
|
+
|
|
335
|
+
#### 特徴
|
|
336
|
+
```plantuml
|
|
337
|
+
@startuml
|
|
338
|
+
package "コマンド側" {
|
|
339
|
+
class ReservationAggregate
|
|
340
|
+
class DomainEvent
|
|
341
|
+
}
|
|
342
|
+
|
|
343
|
+
package "イベントストア" {
|
|
344
|
+
class EventStream
|
|
345
|
+
}
|
|
346
|
+
|
|
347
|
+
package "クエリ側" {
|
|
348
|
+
class ReservationProjection
|
|
349
|
+
class ReadModel
|
|
350
|
+
}
|
|
351
|
+
|
|
352
|
+
ReservationAggregate --> EventStream : イベント保存
|
|
353
|
+
EventStream --> ReservationProjection : イベント再生
|
|
354
|
+
ReservationProjection --> ReadModel : プロジェクション更新
|
|
355
|
+
@enduml
|
|
356
|
+
```
|
|
357
|
+
|
|
358
|
+
**メリット**:
|
|
359
|
+
- 完全な監査証跡
|
|
360
|
+
- 時系列分析が可能
|
|
361
|
+
- 高いスケーラビリティ
|
|
362
|
+
- 複雑なクエリに対応
|
|
363
|
+
|
|
364
|
+
**デメリット**:
|
|
365
|
+
- 実装の複雑さ
|
|
366
|
+
- 高い学習コスト
|
|
367
|
+
- 結果整合性の考慮が必要
|
|
368
|
+
|
|
369
|
+
#### 実装指針
|
|
370
|
+
|
|
371
|
+
```java
|
|
372
|
+
// イベント例
|
|
373
|
+
public class ReservationCreatedEvent extends DomainEvent {
|
|
374
|
+
private final ReservationId reservationId;
|
|
375
|
+
private final UserId userId;
|
|
376
|
+
private final MeetingRoomId roomId;
|
|
377
|
+
private final TimeSlot timeSlot;
|
|
378
|
+
private final LocalDateTime occurredAt;
|
|
379
|
+
}
|
|
380
|
+
|
|
381
|
+
// 集約例
|
|
382
|
+
public class ReservationAggregate {
|
|
383
|
+
private ReservationId id;
|
|
384
|
+
private List<DomainEvent> changes = new ArrayList<>();
|
|
385
|
+
|
|
386
|
+
public void create(UserId userId, MeetingRoomId roomId, TimeSlot timeSlot) {
|
|
387
|
+
var event = new ReservationCreatedEvent(id, userId, roomId, timeSlot, LocalDateTime.now());
|
|
388
|
+
apply(event);
|
|
389
|
+
}
|
|
390
|
+
|
|
391
|
+
private void apply(DomainEvent event) {
|
|
392
|
+
changes.add(event);
|
|
393
|
+
// 状態更新ロジック
|
|
394
|
+
}
|
|
395
|
+
}
|
|
396
|
+
```
|
|
397
|
+
|
|
398
|
+
## アーキテクチャスタイル詳細
|
|
399
|
+
|
|
400
|
+
### 1. レイヤードアーキテクチャ(3層)
|
|
401
|
+
|
|
402
|
+
#### 適用場面
|
|
403
|
+
- **永続化モデル**: 単一
|
|
404
|
+
- **システム特性**: 標準的なエンタープライズアプリケーション
|
|
405
|
+
|
|
406
|
+
#### 構造
|
|
407
|
+
```plantuml
|
|
408
|
+
@startuml
|
|
409
|
+
package "プレゼンテーション層" {
|
|
410
|
+
[コントローラー]
|
|
411
|
+
[UI]
|
|
412
|
+
}
|
|
413
|
+
|
|
414
|
+
package "ビジネスロジック層" {
|
|
415
|
+
[サービス]
|
|
416
|
+
[ドメインモデル]
|
|
417
|
+
}
|
|
418
|
+
|
|
419
|
+
package "データアクセス層" {
|
|
420
|
+
[リポジトリ]
|
|
421
|
+
[DAO]
|
|
422
|
+
}
|
|
423
|
+
|
|
424
|
+
database "データベース"
|
|
425
|
+
|
|
426
|
+
[コントローラー] --> [サービス]
|
|
427
|
+
[UI] --> [コントローラー]
|
|
428
|
+
[サービス] --> [リポジトリ]
|
|
429
|
+
[リポジトリ] --> [データベース]
|
|
430
|
+
@enduml
|
|
431
|
+
```
|
|
432
|
+
|
|
433
|
+
#### 実装指針
|
|
434
|
+
|
|
435
|
+
```java
|
|
436
|
+
// プレゼンテーション層
|
|
437
|
+
@RestController
|
|
438
|
+
public class ReservationController {
|
|
439
|
+
private final ReservationService service;
|
|
440
|
+
|
|
441
|
+
@PostMapping("/reservations")
|
|
442
|
+
public ResponseEntity<Reservation> create(@RequestBody CreateReservationRequest request) {
|
|
443
|
+
var reservation = service.createReservation(request);
|
|
444
|
+
return ResponseEntity.ok(reservation);
|
|
445
|
+
}
|
|
446
|
+
}
|
|
447
|
+
|
|
448
|
+
// ビジネスロジック層
|
|
449
|
+
@Service
|
|
450
|
+
public class ReservationService {
|
|
451
|
+
private final ReservationRepository repository;
|
|
452
|
+
|
|
453
|
+
public Reservation createReservation(CreateReservationRequest request) {
|
|
454
|
+
// ビジネスロジック実行
|
|
455
|
+
var reservation = new Reservation(request);
|
|
456
|
+
return repository.save(reservation);
|
|
457
|
+
}
|
|
458
|
+
}
|
|
459
|
+
|
|
460
|
+
// データアクセス層
|
|
461
|
+
@Repository
|
|
462
|
+
public interface ReservationRepository extends JpaRepository<Reservation, Long> {
|
|
463
|
+
List<Reservation> findByUserId(Long userId);
|
|
464
|
+
}
|
|
465
|
+
```
|
|
466
|
+
|
|
467
|
+
### 2. レイヤードアーキテクチャ(4層)
|
|
468
|
+
|
|
469
|
+
#### 適用場面
|
|
470
|
+
- **永続化モデル**: 複数
|
|
471
|
+
- **システム特性**: 大規模エンタープライズアプリケーション
|
|
472
|
+
|
|
473
|
+
#### 構造
|
|
474
|
+
```plantuml
|
|
475
|
+
@startuml
|
|
476
|
+
package "プレゼンテーション層" {
|
|
477
|
+
[コントローラー]
|
|
478
|
+
}
|
|
479
|
+
|
|
480
|
+
package "アプリケーション層" {
|
|
481
|
+
[アプリケーションサービス]
|
|
482
|
+
[ユースケース]
|
|
483
|
+
}
|
|
484
|
+
|
|
485
|
+
package "ドメイン層" {
|
|
486
|
+
[ドメインサービス]
|
|
487
|
+
[エンティティ]
|
|
488
|
+
}
|
|
489
|
+
|
|
490
|
+
package "インフラストラクチャ層" {
|
|
491
|
+
[リポジトリ実装]
|
|
492
|
+
[外部API]
|
|
493
|
+
}
|
|
494
|
+
|
|
495
|
+
[コントローラー] --> [アプリケーションサービス]
|
|
496
|
+
[アプリケーションサービス] --> [ドメインサービス]
|
|
497
|
+
[アプリケーションサービス] --> [リポジトリ実装]
|
|
498
|
+
@enduml
|
|
499
|
+
```
|
|
500
|
+
|
|
501
|
+
#### レイヤー責務
|
|
502
|
+
|
|
503
|
+
| レイヤー | 責務 |
|
|
504
|
+
|----------|------|
|
|
505
|
+
| **プレゼンテーション** | HTTP要求/応答、入力検証、認証 |
|
|
506
|
+
| **アプリケーション** | ユースケース制御、トランザクション境界 |
|
|
507
|
+
| **ドメイン** | ビジネスルール、不変条件、ドメインサービス |
|
|
508
|
+
| **インフラストラクチャ** | 外部システム連携、永続化、技術的関心事 |
|
|
509
|
+
|
|
510
|
+
### 3. ポートとアダプターアーキテクチャ(ヘキサゴナル)
|
|
511
|
+
|
|
512
|
+
#### 適用場面
|
|
513
|
+
- **ドメインモデル**: あり
|
|
514
|
+
- **永続化モデル**: 単一
|
|
515
|
+
- **システム特性**: マイクロサービス、高いテスト容易性が要求される
|
|
516
|
+
|
|
517
|
+
#### 構造
|
|
518
|
+
```plantuml
|
|
519
|
+
@startuml
|
|
520
|
+
hexagon "ドメイン\nモデル" as domain
|
|
521
|
+
|
|
522
|
+
interface "入力ポート" as inport
|
|
523
|
+
interface "出力ポート" as outport
|
|
524
|
+
|
|
525
|
+
[Webアダプター] as web
|
|
526
|
+
[CLIアダプター] as cli
|
|
527
|
+
[データベースアダプター] as db
|
|
528
|
+
[外部APIアダプター] as api
|
|
529
|
+
|
|
530
|
+
web --> inport
|
|
531
|
+
cli --> inport
|
|
532
|
+
inport --> domain
|
|
533
|
+
domain --> outport
|
|
534
|
+
outport <|-- db
|
|
535
|
+
outport <|-- api
|
|
536
|
+
@enduml
|
|
537
|
+
```
|
|
538
|
+
|
|
539
|
+
#### 実装指針
|
|
540
|
+
|
|
541
|
+
```java
|
|
542
|
+
// 入力ポート(ユースケース)
|
|
543
|
+
public interface CreateReservationUseCase {
|
|
544
|
+
ReservationId execute(CreateReservationCommand command);
|
|
545
|
+
}
|
|
546
|
+
|
|
547
|
+
// 出力ポート(リポジトリインターフェース)
|
|
548
|
+
public interface ReservationRepository {
|
|
549
|
+
void save(Reservation reservation);
|
|
550
|
+
Optional<Reservation> findById(ReservationId id);
|
|
551
|
+
}
|
|
552
|
+
|
|
553
|
+
// ドメインサービス(実装)
|
|
554
|
+
@Service
|
|
555
|
+
public class ReservationService implements CreateReservationUseCase {
|
|
556
|
+
private final ReservationRepository repository;
|
|
557
|
+
|
|
558
|
+
@Override
|
|
559
|
+
public ReservationId execute(CreateReservationCommand command) {
|
|
560
|
+
var reservation = Reservation.create(command);
|
|
561
|
+
repository.save(reservation);
|
|
562
|
+
return reservation.getId();
|
|
563
|
+
}
|
|
564
|
+
}
|
|
565
|
+
|
|
566
|
+
// アダプター(インフラストラクチャ)
|
|
567
|
+
@Repository
|
|
568
|
+
public class JpaReservationRepository implements ReservationRepository {
|
|
569
|
+
private final SpringDataReservationRepository jpaRepository;
|
|
570
|
+
|
|
571
|
+
@Override
|
|
572
|
+
public void save(Reservation reservation) {
|
|
573
|
+
var entity = toEntity(reservation);
|
|
574
|
+
jpaRepository.save(entity);
|
|
575
|
+
}
|
|
576
|
+
}
|
|
577
|
+
```
|
|
578
|
+
|
|
579
|
+
### 4. CQRSアーキテクチャ
|
|
580
|
+
|
|
581
|
+
#### 適用場面
|
|
582
|
+
- **永続化モデル**: 複数
|
|
583
|
+
- **特殊要件**: イベント履歴式、高スケーラビリティ
|
|
584
|
+
- **システム特性**: 複雑なクエリ要件、読み書き分離
|
|
585
|
+
|
|
586
|
+
#### 構造
|
|
587
|
+
```plantuml
|
|
588
|
+
@startuml
|
|
589
|
+
package "コマンド側" {
|
|
590
|
+
[コマンドハンドラー] as ch
|
|
591
|
+
[書き込みモデル] as wm
|
|
592
|
+
[コマンドストア] as cs
|
|
593
|
+
}
|
|
594
|
+
|
|
595
|
+
package "クエリ側" {
|
|
596
|
+
[クエリハンドラー] as qh
|
|
597
|
+
[読み取りモデル] as rm
|
|
598
|
+
[クエリストア] as qs
|
|
599
|
+
}
|
|
600
|
+
|
|
601
|
+
package "イベント" {
|
|
602
|
+
[イベントバス] as eb
|
|
603
|
+
}
|
|
604
|
+
|
|
605
|
+
[コマンド] --> ch
|
|
606
|
+
ch --> wm
|
|
607
|
+
wm --> cs
|
|
608
|
+
wm --> eb
|
|
609
|
+
eb --> rm
|
|
610
|
+
rm --> qs
|
|
611
|
+
[クエリ] --> qh
|
|
612
|
+
qh --> rm
|
|
613
|
+
@enduml
|
|
614
|
+
```
|
|
615
|
+
|
|
616
|
+
#### 実装指針
|
|
617
|
+
|
|
618
|
+
```java
|
|
619
|
+
// コマンド側
|
|
620
|
+
@CommandHandler
|
|
621
|
+
public class CreateReservationHandler {
|
|
622
|
+
private final ReservationWriteRepository writeRepository;
|
|
623
|
+
private final EventBus eventBus;
|
|
624
|
+
|
|
625
|
+
public void handle(CreateReservationCommand command) {
|
|
626
|
+
var reservation = new ReservationAggregate(command);
|
|
627
|
+
writeRepository.save(reservation);
|
|
628
|
+
|
|
629
|
+
var events = reservation.getUncommittedEvents();
|
|
630
|
+
eventBus.publishAll(events);
|
|
631
|
+
}
|
|
632
|
+
}
|
|
633
|
+
|
|
634
|
+
// クエリ側
|
|
635
|
+
@QueryHandler
|
|
636
|
+
public class ReservationQueryHandler {
|
|
637
|
+
private final ReservationReadRepository readRepository;
|
|
638
|
+
|
|
639
|
+
public List<ReservationView> handle(FindReservationsByUserQuery query) {
|
|
640
|
+
return readRepository.findByUserId(query.getUserId());
|
|
641
|
+
}
|
|
642
|
+
}
|
|
643
|
+
|
|
644
|
+
// イベントハンドラー(プロジェクション更新)
|
|
645
|
+
@EventHandler
|
|
646
|
+
public class ReservationProjectionHandler {
|
|
647
|
+
private final ReservationReadRepository readRepository;
|
|
648
|
+
|
|
649
|
+
public void on(ReservationCreatedEvent event) {
|
|
650
|
+
var view = new ReservationView(event);
|
|
651
|
+
readRepository.save(view);
|
|
652
|
+
}
|
|
653
|
+
}
|
|
654
|
+
```
|
|
655
|
+
|
|
656
|
+
## テスト戦略とアーキテクチャの関係
|
|
657
|
+
|
|
658
|
+
### ピラミッド形テスト(ドメインモデル・イベント履歴式)
|
|
659
|
+
|
|
660
|
+
```plantuml
|
|
661
|
+
@startuml
|
|
662
|
+
rectangle "E2Eテスト\n(少数)" as e2e #lightcoral
|
|
663
|
+
rectangle "統合テスト\n(中程度)" as integration #orange
|
|
664
|
+
rectangle "ユニットテスト\n(多数)" as unit #lightgreen
|
|
665
|
+
|
|
666
|
+
unit -up-> integration
|
|
667
|
+
integration -up-> e2e
|
|
668
|
+
@enduml
|
|
669
|
+
```
|
|
670
|
+
|
|
671
|
+
**特徴**:
|
|
672
|
+
- ユニットテスト中心(80%)
|
|
673
|
+
- ドメインロジックの徹底検証
|
|
674
|
+
- 高品質なビジネスルール実装
|
|
675
|
+
|
|
676
|
+
### ダイヤモンド形テスト(アクティブレコード)
|
|
677
|
+
|
|
678
|
+
```plantuml
|
|
679
|
+
@startuml
|
|
680
|
+
rectangle "E2Eテスト\n(中程度)" as e2e #orange
|
|
681
|
+
rectangle "統合テスト\n(多数)" as integration #lightgreen
|
|
682
|
+
rectangle "ユニットテスト\n(中程度)" as unit #orange
|
|
683
|
+
|
|
684
|
+
unit -up-> integration
|
|
685
|
+
integration -up-> e2e
|
|
686
|
+
@enduml
|
|
687
|
+
```
|
|
688
|
+
|
|
689
|
+
**特徴**:
|
|
690
|
+
- 統合テスト中心(60%)
|
|
691
|
+
- データアクセスロジックの重点検証
|
|
692
|
+
- ORMとの連携確認
|
|
693
|
+
|
|
694
|
+
### 逆ピラミッド形テスト(トランザクションスクリプト)
|
|
695
|
+
|
|
696
|
+
```plantuml
|
|
697
|
+
@startuml
|
|
698
|
+
rectangle "E2Eテスト\n(多数)" as e2e #lightgreen
|
|
699
|
+
rectangle "統合テスト\n(中程度)" as integration #orange
|
|
700
|
+
rectangle "ユニットテスト\n(少数)" as unit #lightcoral
|
|
701
|
+
|
|
702
|
+
unit -up-> integration
|
|
703
|
+
integration -up-> e2e
|
|
704
|
+
@enduml
|
|
705
|
+
```
|
|
706
|
+
|
|
707
|
+
**特徴**:
|
|
708
|
+
- E2Eテスト中心(70%)
|
|
709
|
+
- エンドツーエンドの動作確認
|
|
710
|
+
- シナリオベースの検証
|
|
711
|
+
|
|
712
|
+
## アーキテクチャ評価基準
|
|
713
|
+
|
|
714
|
+
### 品質属性評価マトリックス
|
|
715
|
+
|
|
716
|
+
| 品質属性 | トランザクション<br>スクリプト | アクティブ<br>レコード | ドメイン<br>モデル | イベント履歴式<br>ドメインモデル |
|
|
717
|
+
|----------|:---:|:---:|:---:|:---:|
|
|
718
|
+
| **開発速度** | ⭐⭐⭐⭐⭐ | ⭐⭐⭐⭐ | ⭐⭐ | ⭐ |
|
|
719
|
+
| **保守性** | ⭐⭐ | ⭐⭐⭐ | ⭐⭐⭐⭐⭐ | ⭐⭐⭐⭐ |
|
|
720
|
+
| **テスト容易性** | ⭐⭐ | ⭐⭐ | ⭐⭐⭐⭐⭐ | ⭐⭐⭐⭐ |
|
|
721
|
+
| **スケーラビリティ** | ⭐⭐ | ⭐⭐⭐ | ⭐⭐⭐⭐ | ⭐⭐⭐⭐⭐ |
|
|
722
|
+
| **複雑さ対応** | ⭐ | ⭐⭐ | ⭐⭐⭐⭐⭐ | ⭐⭐⭐⭐⭐ |
|
|
723
|
+
| **監査要件** | ⭐ | ⭐⭐ | ⭐⭐⭐ | ⭐⭐⭐⭐⭐ |
|
|
724
|
+
|
|
725
|
+
### アーキテクチャスタイル評価マトリックス
|
|
726
|
+
|
|
727
|
+
| 品質属性 | レイヤード<br>3層 | レイヤード<br>4層 | ポートと<br>アダプター | CQRS |
|
|
728
|
+
|----------|:---:|:---:|:---:|:---:|
|
|
729
|
+
| **学習容易性** | ⭐⭐⭐⭐⭐ | ⭐⭐⭐⭐ | ⭐⭐⭐ | ⭐⭐ |
|
|
730
|
+
| **テスト容易性** | ⭐⭐⭐ | ⭐⭐⭐⭐ | ⭐⭐⭐⭐⭐ | ⭐⭐⭐⭐ |
|
|
731
|
+
| **関心事分離** | ⭐⭐⭐ | ⭐⭐⭐⭐ | ⭐⭐⭐⭐⭐ | ⭐⭐⭐⭐⭐ |
|
|
732
|
+
| **パフォーマンス** | ⭐⭐⭐ | ⭐⭐⭐ | ⭐⭐⭐ | ⭐⭐⭐⭐⭐ |
|
|
733
|
+
| **スケーラビリティ** | ⭐⭐ | ⭐⭐⭐ | ⭐⭐⭐⭐ | ⭐⭐⭐⭐⭐ |
|
|
734
|
+
| **進化性** | ⭐⭐ | ⭐⭐⭐ | ⭐⭐⭐⭐⭐ | ⭐⭐⭐⭐ |
|
|
735
|
+
|
|
736
|
+
## 実装ベストプラクティス
|
|
737
|
+
|
|
738
|
+
### 1. 依存関係管理
|
|
739
|
+
|
|
740
|
+
```java
|
|
741
|
+
// ✅ 良い例:インターフェースに依存
|
|
742
|
+
public class ReservationService {
|
|
743
|
+
private final ReservationRepository repository; // インターフェース
|
|
744
|
+
private final NotificationService notificationService; // インターフェース
|
|
745
|
+
}
|
|
746
|
+
|
|
747
|
+
// ❌ 悪い例:具象クラスに依存
|
|
748
|
+
public class ReservationService {
|
|
749
|
+
private final JpaReservationRepository repository; // 具象クラス
|
|
750
|
+
private final EmailService emailService; // 具象クラス
|
|
751
|
+
}
|
|
752
|
+
```
|
|
753
|
+
|
|
754
|
+
### 2. レイヤー分離
|
|
755
|
+
|
|
756
|
+
```java
|
|
757
|
+
// ✅ 良い例:適切なレイヤー分離
|
|
758
|
+
@RestController
|
|
759
|
+
public class ReservationController {
|
|
760
|
+
@PostMapping("/reservations")
|
|
761
|
+
public ResponseEntity<ReservationResponse> create(@RequestBody CreateReservationRequest request) {
|
|
762
|
+
// プレゼンテーション層の責務のみ
|
|
763
|
+
var command = toCommand(request);
|
|
764
|
+
var result = useCase.execute(command);
|
|
765
|
+
return ResponseEntity.ok(toResponse(result));
|
|
766
|
+
}
|
|
767
|
+
}
|
|
768
|
+
|
|
769
|
+
// ❌ 悪い例:レイヤー違反
|
|
770
|
+
@RestController
|
|
771
|
+
public class ReservationController {
|
|
772
|
+
@PostMapping("/reservations")
|
|
773
|
+
public ResponseEntity<ReservationResponse> create(@RequestBody CreateReservationRequest request) {
|
|
774
|
+
// ビジネスロジックをコントローラーに記述(レイヤー違反)
|
|
775
|
+
if (request.getStartTime().isBefore(LocalDateTime.now().plusHours(2))) {
|
|
776
|
+
throw new ValidationException("2時間前までの予約が必要です");
|
|
777
|
+
}
|
|
778
|
+
// ...
|
|
779
|
+
}
|
|
780
|
+
}
|
|
781
|
+
```
|
|
782
|
+
|
|
783
|
+
### 3. エラーハンドリング
|
|
784
|
+
|
|
785
|
+
```java
|
|
786
|
+
// ✅ 良い例:適切なエラー伝播
|
|
787
|
+
public class ReservationService {
|
|
788
|
+
public ReservationId createReservation(CreateReservationCommand command) {
|
|
789
|
+
try {
|
|
790
|
+
var reservation = Reservation.create(command);
|
|
791
|
+
repository.save(reservation);
|
|
792
|
+
return reservation.getId();
|
|
793
|
+
} catch (DomainException e) {
|
|
794
|
+
// ドメイン例外は適切にハンドリング
|
|
795
|
+
throw new ReservationException("予約作成に失敗しました", e);
|
|
796
|
+
}
|
|
797
|
+
}
|
|
798
|
+
}
|
|
799
|
+
|
|
800
|
+
// ❌ 悪い例:例外の隠蔽
|
|
801
|
+
public class ReservationService {
|
|
802
|
+
public ReservationId createReservation(CreateReservationCommand command) {
|
|
803
|
+
try {
|
|
804
|
+
var reservation = Reservation.create(command);
|
|
805
|
+
repository.save(reservation);
|
|
806
|
+
return reservation.getId();
|
|
807
|
+
} catch (Exception e) {
|
|
808
|
+
// すべての例外を隠蔽(問題の特定が困難)
|
|
809
|
+
return null;
|
|
810
|
+
}
|
|
811
|
+
}
|
|
812
|
+
}
|
|
813
|
+
```
|
|
814
|
+
|
|
815
|
+
# フロントエンドアーキテクチャ
|
|
816
|
+
|
|
817
|
+
## フロントエンドアーキテクチャ決定フロー
|
|
818
|
+
|
|
819
|
+
### 主要アーキテクチャ決定ポイント
|
|
820
|
+
|
|
821
|
+
```plantuml
|
|
822
|
+
@startuml
|
|
823
|
+
start
|
|
824
|
+
|
|
825
|
+
:プロジェクト分析;
|
|
826
|
+
|
|
827
|
+
if (プロジェクト規模) then (大規模)
|
|
828
|
+
:11フォルダ標準構造;
|
|
829
|
+
note right
|
|
830
|
+
components/config/features/layouts/
|
|
831
|
+
lib/pages/providers/stores/
|
|
832
|
+
testing/types/utils/
|
|
833
|
+
end note
|
|
834
|
+
else (中小規模)
|
|
835
|
+
:6フォルダシンプル構造;
|
|
836
|
+
note right
|
|
837
|
+
components/pages/hooks/
|
|
838
|
+
utils/types/config/
|
|
839
|
+
end note
|
|
840
|
+
endif
|
|
841
|
+
|
|
842
|
+
if (レンダリング戦略) then (SEO重要)
|
|
843
|
+
if (更新頻度) then (高頻度)
|
|
844
|
+
:サーバーサイドレンダリング;
|
|
845
|
+
note right: Next.js
|
|
846
|
+
else (低頻度)
|
|
847
|
+
:静的生成;
|
|
848
|
+
note right: Gatsby/Next.js SSG
|
|
849
|
+
endif
|
|
850
|
+
else (内部ツール・ダッシュボード)
|
|
851
|
+
:SPA;
|
|
852
|
+
note right: React SPA
|
|
853
|
+
endif
|
|
854
|
+
|
|
855
|
+
if (状態管理複雑性) then (複雑)
|
|
856
|
+
if (更新頻度) then (高頻度)
|
|
857
|
+
:Atom系(Recoil/Jotai);
|
|
858
|
+
note right: stores/フォルダに集約
|
|
859
|
+
else (共有多い)
|
|
860
|
+
:Redux with RTK;
|
|
861
|
+
note right: stores/フォルダに集約
|
|
862
|
+
endif
|
|
863
|
+
else (シンプル)
|
|
864
|
+
:Context API + hooks / Zustand;
|
|
865
|
+
note right: providers/フォルダに集約
|
|
866
|
+
endif
|
|
867
|
+
|
|
868
|
+
if (パフォーマンス重要度) then (高)
|
|
869
|
+
:CSS/SCSS/Tailwind;
|
|
870
|
+
note right: 頻繁レンダリング対応
|
|
871
|
+
else (開発速度重視)
|
|
872
|
+
:CSS-in-JS (Styled Components);
|
|
873
|
+
note right: components/内で管理
|
|
874
|
+
endif
|
|
875
|
+
|
|
876
|
+
stop
|
|
877
|
+
@enduml
|
|
878
|
+
```
|
|
879
|
+
|
|
880
|
+
## フロントエンド設計原則
|
|
881
|
+
|
|
882
|
+
### 良いアーキテクチャ決定
|
|
883
|
+
|
|
884
|
+
#### 1. **プロジェクト構造**
|
|
885
|
+
|
|
886
|
+
##### 標準的なフロントエンドプロジェクト構造
|
|
887
|
+
```
|
|
888
|
+
src/
|
|
889
|
+
├── components/ # (1) 共通UIコンポーネント
|
|
890
|
+
│ ├── ui/ # 基本UIパーツ(Button, Input, Modal等)
|
|
891
|
+
│ └── layout/ # レイアウトコンポーネント
|
|
892
|
+
├── config/ # (2) アプリケーション設定
|
|
893
|
+
│ ├── constants.ts # 定数定義
|
|
894
|
+
│ ├── env.ts # 環境変数管理
|
|
895
|
+
│ └── api.ts # API設定
|
|
896
|
+
├── features/ # (3) フィーチャー単位のコンポーネント
|
|
897
|
+
│ ├── auth/ # 認証機能
|
|
898
|
+
│ ├── reservation/ # 予約機能
|
|
899
|
+
│ └── meeting-room/# 会議室管理機能
|
|
900
|
+
├── layouts/ # (4) アプリケーションレイアウト
|
|
901
|
+
│ ├── AppLayout.tsx
|
|
902
|
+
│ ├── AuthLayout.tsx
|
|
903
|
+
│ └── DashboardLayout.tsx
|
|
904
|
+
├── lib/ # (5) 外部ライブラリ設定・ユーティリティ
|
|
905
|
+
│ ├── auth.ts # 認証ライブラリ設定
|
|
906
|
+
│ ├── api-client.ts# API クライアント
|
|
907
|
+
│ └── utils.ts # ヘルパー関数
|
|
908
|
+
├── pages/ # (6) ページコンポーネント(ルーティング)
|
|
909
|
+
│ ├── LoginPage.tsx
|
|
910
|
+
│ ├── DashboardPage.tsx
|
|
911
|
+
│ └── ReservationPage.tsx
|
|
912
|
+
├── providers/ # (7) Context Provider
|
|
913
|
+
│ ├── AuthProvider.tsx
|
|
914
|
+
│ ├── ThemeProvider.tsx
|
|
915
|
+
│ └── AppProviders.tsx
|
|
916
|
+
├── stores/ # (8) 状態管理
|
|
917
|
+
│ ├── authStore.ts
|
|
918
|
+
│ ├── uiStore.ts
|
|
919
|
+
│ └── index.ts
|
|
920
|
+
├── testing/ # (9) テスト用ユーティリティ
|
|
921
|
+
│ ├── setup.ts # テスト環境設定
|
|
922
|
+
│ ├── mocks/ # モックデータ
|
|
923
|
+
│ └── utils.ts # テストヘルパー
|
|
924
|
+
├── types/ # (10) TypeScript型定義
|
|
925
|
+
│ ├── api.ts # API型定義
|
|
926
|
+
│ ├── auth.ts # 認証関連型
|
|
927
|
+
│ └── common.ts # 共通型定義
|
|
928
|
+
└── utils/ # (11) 汎用ユーティリティ関数
|
|
929
|
+
├── format.ts # フォーマット関数
|
|
930
|
+
├── validation.ts# バリデーション
|
|
931
|
+
└── date.ts # 日付操作
|
|
932
|
+
```
|
|
933
|
+
|
|
934
|
+
##### プロジェクト規模による使い分け
|
|
935
|
+
|
|
936
|
+
**小規模プロジェクト (シンプル構造)**:
|
|
937
|
+
```
|
|
938
|
+
src/
|
|
939
|
+
├── components/ # UI コンポーネント
|
|
940
|
+
├── pages/ # ページコンポーネント
|
|
941
|
+
├── hooks/ # カスタムフック
|
|
942
|
+
├── utils/ # ユーティリティ
|
|
943
|
+
├── types/ # 型定義
|
|
944
|
+
└── config/ # 設定
|
|
945
|
+
```
|
|
946
|
+
|
|
947
|
+
**大規模プロジェクト (フル構造)**:
|
|
948
|
+
上記の標準11フォルダ構成を使用
|
|
949
|
+
|
|
950
|
+
#### 2. **コンポーネント設計**
|
|
951
|
+
- **小さなコンポーネント**: 単一責任の原則に従った分割
|
|
952
|
+
- **関心事の分離**: UI・ロジック・状態管理の明確な分離
|
|
953
|
+
- **再利用性**: 汎用コンポーネントの適切な抽象化
|
|
954
|
+
|
|
955
|
+
#### 3. **状態管理戦略**
|
|
956
|
+
|
|
957
|
+
```plantuml
|
|
958
|
+
@startuml
|
|
959
|
+
package "状態管理レイヤー" {
|
|
960
|
+
rectangle "サーバー状態" as server #lightblue
|
|
961
|
+
rectangle "クライアント状態" as client #lightgreen
|
|
962
|
+
rectangle "UI状態" as ui #lightyellow
|
|
963
|
+
}
|
|
964
|
+
|
|
965
|
+
package "管理ツール" {
|
|
966
|
+
rectangle "React Query / SWR" as rq
|
|
967
|
+
rectangle "Redux / Zustand" as global
|
|
968
|
+
rectangle "useState / useReducer" as local
|
|
969
|
+
}
|
|
970
|
+
|
|
971
|
+
server --> rq : キャッシュ・同期
|
|
972
|
+
client --> global : グローバル共有
|
|
973
|
+
ui --> local : ローカル管理
|
|
974
|
+
@enduml
|
|
975
|
+
```
|
|
976
|
+
|
|
977
|
+
#### 4. **レンダリング戦略**
|
|
978
|
+
|
|
979
|
+
| 戦略 | 適用場面 | 特徴 | 例 |
|
|
980
|
+
|------|----------|------|-----|
|
|
981
|
+
| **SPA** | ダッシュボード・管理画面 | 高いインタラクティビティ | React |
|
|
982
|
+
| **SSR** | SEO重要・高更新頻度 | 初期表示高速・SEO対応 | Next.js |
|
|
983
|
+
| **SSG** | SEO重要・低更新頻度 | 最高のパフォーマンス | Gatsby |
|
|
984
|
+
|
|
985
|
+
### 悪いアーキテクチャ決定
|
|
986
|
+
|
|
987
|
+
#### ❌ **避けるべき設計**
|
|
988
|
+
|
|
989
|
+
1. **フラットなプロジェクト構造**
|
|
990
|
+
```
|
|
991
|
+
src/
|
|
992
|
+
├── LoginPage.tsx # 管理困難
|
|
993
|
+
├── Button.tsx # コンポーネント分類なし
|
|
994
|
+
├── UserService.ts # ロジック散在
|
|
995
|
+
├── constants.ts # 設定管理なし
|
|
996
|
+
├── utils.ts # 目的不明
|
|
997
|
+
├── types.ts # 型定義混在
|
|
998
|
+
└── ... # 数十個のファイル(役割不明)
|
|
999
|
+
```
|
|
1000
|
+
|
|
1001
|
+
2. **不適切な責務分散**
|
|
1002
|
+
```
|
|
1003
|
+
src/
|
|
1004
|
+
├── components/ # 全コンポーネント混在
|
|
1005
|
+
│ ├── LoginPage.tsx # ページなのにcomponents内
|
|
1006
|
+
│ ├── Button.tsx # UIコンポーネント
|
|
1007
|
+
│ ├── AuthProvider.tsx # Providerなのにcomponents内
|
|
1008
|
+
│ └── authStore.ts # ストアなのにcomponents内
|
|
1009
|
+
└── utils/ # 何でも入れる場所
|
|
1010
|
+
├── api.ts # API設定
|
|
1011
|
+
├── constants.ts # 定数
|
|
1012
|
+
├── types.ts # 型定義
|
|
1013
|
+
└── validation.ts # バリデーション
|
|
1014
|
+
```
|
|
1015
|
+
|
|
1016
|
+
2. **巨大で密結合なコンポーネント**
|
|
1017
|
+
```tsx
|
|
1018
|
+
// ❌ 悪い例
|
|
1019
|
+
function MegaComponent() {
|
|
1020
|
+
// 500行以上のコンポーネント
|
|
1021
|
+
// 複数の責務を持つ
|
|
1022
|
+
// テストが困難
|
|
1023
|
+
}
|
|
1024
|
+
```
|
|
1025
|
+
|
|
1026
|
+
3. **不適切な状態管理**
|
|
1027
|
+
```tsx
|
|
1028
|
+
// ❌ 悪い例:すべてをグローバル状態に
|
|
1029
|
+
const globalState = {
|
|
1030
|
+
user: {},
|
|
1031
|
+
reservations: [],
|
|
1032
|
+
modalOpen: false, // UI状態もグローバル化
|
|
1033
|
+
buttonColor: 'blue' // 一時的な状態もグローバル化
|
|
1034
|
+
}
|
|
1035
|
+
```
|
|
1036
|
+
|
|
1037
|
+
4. **セキュリティの軽視**
|
|
1038
|
+
```tsx
|
|
1039
|
+
// ❌ 悪い例:入力値のサニタイジング不備
|
|
1040
|
+
function UserInput({ input }) {
|
|
1041
|
+
return <div dangerouslySetInnerHTML={{__html: input}} />
|
|
1042
|
+
}
|
|
1043
|
+
```
|
|
1044
|
+
|
|
1045
|
+
## フロントエンド実装パターン
|
|
1046
|
+
|
|
1047
|
+
### 1. **コンポーネント設計パターン**
|
|
1048
|
+
|
|
1049
|
+
#### Container/Presentational パターン
|
|
1050
|
+
|
|
1051
|
+
```
|
|
1052
|
+
// ファイル配置例
|
|
1053
|
+
src/
|
|
1054
|
+
├── features/reservation/
|
|
1055
|
+
│ ├── components/
|
|
1056
|
+
│ │ ├── ReservationContainer.tsx # Container
|
|
1057
|
+
│ │ └── ReservationList.tsx # Presentational
|
|
1058
|
+
│ ├── hooks/
|
|
1059
|
+
│ │ └── useReservations.ts # Custom Hook
|
|
1060
|
+
│ └── types/
|
|
1061
|
+
│ └── reservation.ts # 型定義
|
|
1062
|
+
```
|
|
1063
|
+
|
|
1064
|
+
```tsx
|
|
1065
|
+
// ✅ Container: ロジック担当 (features/reservation/components/ReservationContainer.tsx)
|
|
1066
|
+
import { useReservations } from '../hooks/useReservations';
|
|
1067
|
+
import { ReservationList } from './ReservationList';
|
|
1068
|
+
|
|
1069
|
+
export function ReservationContainer() {
|
|
1070
|
+
const { reservations, loading, createReservation } = useReservations();
|
|
1071
|
+
const handleCreate = useCallback((data) => {
|
|
1072
|
+
createReservation(data);
|
|
1073
|
+
}, [createReservation]);
|
|
1074
|
+
|
|
1075
|
+
return (
|
|
1076
|
+
<ReservationList
|
|
1077
|
+
reservations={reservations}
|
|
1078
|
+
loading={loading}
|
|
1079
|
+
onCreate={handleCreate}
|
|
1080
|
+
/>
|
|
1081
|
+
);
|
|
1082
|
+
}
|
|
1083
|
+
|
|
1084
|
+
// ✅ Presentational: UI担当 (features/reservation/components/ReservationList.tsx)
|
|
1085
|
+
import { LoadingSpinner } from '@/components/ui/LoadingSpinner';
|
|
1086
|
+
import { CreateButton } from '@/components/ui/CreateButton';
|
|
1087
|
+
import type { Reservation } from '../types/reservation';
|
|
1088
|
+
|
|
1089
|
+
interface Props {
|
|
1090
|
+
reservations: Reservation[];
|
|
1091
|
+
loading: boolean;
|
|
1092
|
+
onCreate: (data: CreateReservationData) => void;
|
|
1093
|
+
}
|
|
1094
|
+
|
|
1095
|
+
export function ReservationList({ reservations, loading, onCreate }: Props) {
|
|
1096
|
+
if (loading) return <LoadingSpinner />;
|
|
1097
|
+
|
|
1098
|
+
return (
|
|
1099
|
+
<div>
|
|
1100
|
+
{reservations.map(reservation =>
|
|
1101
|
+
<ReservationItem key={reservation.id} data={reservation} />
|
|
1102
|
+
)}
|
|
1103
|
+
<CreateButton onClick={onCreate} />
|
|
1104
|
+
</div>
|
|
1105
|
+
);
|
|
1106
|
+
}
|
|
1107
|
+
```
|
|
1108
|
+
|
|
1109
|
+
#### Custom Hooks パターン
|
|
1110
|
+
```tsx
|
|
1111
|
+
// ✅ ビジネスロジックの分離
|
|
1112
|
+
function useReservation() {
|
|
1113
|
+
const [reservations, setReservations] = useState([]);
|
|
1114
|
+
const [loading, setLoading] = useState(false);
|
|
1115
|
+
|
|
1116
|
+
const createReservation = useCallback(async (data) => {
|
|
1117
|
+
setLoading(true);
|
|
1118
|
+
try {
|
|
1119
|
+
const result = await api.createReservation(data);
|
|
1120
|
+
setReservations(prev => [...prev, result]);
|
|
1121
|
+
} catch (error) {
|
|
1122
|
+
handleError(error);
|
|
1123
|
+
} finally {
|
|
1124
|
+
setLoading(false);
|
|
1125
|
+
}
|
|
1126
|
+
}, []);
|
|
1127
|
+
|
|
1128
|
+
return { reservations, loading, createReservation };
|
|
1129
|
+
}
|
|
1130
|
+
```
|
|
1131
|
+
|
|
1132
|
+
### 2. **状態管理パターン**
|
|
1133
|
+
|
|
1134
|
+
#### Context + Reducer パターン
|
|
1135
|
+
```tsx
|
|
1136
|
+
// ✅ 状態とアクションの型安全な管理
|
|
1137
|
+
interface AppState {
|
|
1138
|
+
user: User | null;
|
|
1139
|
+
theme: 'light' | 'dark';
|
|
1140
|
+
}
|
|
1141
|
+
|
|
1142
|
+
type AppAction =
|
|
1143
|
+
| { type: 'SET_USER'; payload: User }
|
|
1144
|
+
| { type: 'SET_THEME'; payload: 'light' | 'dark' };
|
|
1145
|
+
|
|
1146
|
+
function appReducer(state: AppState, action: AppAction): AppState {
|
|
1147
|
+
switch (action.type) {
|
|
1148
|
+
case 'SET_USER':
|
|
1149
|
+
return { ...state, user: action.payload };
|
|
1150
|
+
case 'SET_THEME':
|
|
1151
|
+
return { ...state, theme: action.payload };
|
|
1152
|
+
default:
|
|
1153
|
+
return state;
|
|
1154
|
+
}
|
|
1155
|
+
}
|
|
1156
|
+
|
|
1157
|
+
const AppContext = createContext<{
|
|
1158
|
+
state: AppState;
|
|
1159
|
+
dispatch: Dispatch<AppAction>;
|
|
1160
|
+
} | null>(null);
|
|
1161
|
+
```
|
|
1162
|
+
|
|
1163
|
+
### 3. **パフォーマンス最適化パターン**
|
|
1164
|
+
|
|
1165
|
+
#### 遅延読み込み
|
|
1166
|
+
```tsx
|
|
1167
|
+
// ✅ コード分割
|
|
1168
|
+
const ReservationPage = lazy(() => import('./pages/ReservationPage'));
|
|
1169
|
+
const UserManagementPage = lazy(() => import('./pages/UserManagementPage'));
|
|
1170
|
+
|
|
1171
|
+
function App() {
|
|
1172
|
+
return (
|
|
1173
|
+
<Router>
|
|
1174
|
+
<Suspense fallback={<LoadingSpinner />}>
|
|
1175
|
+
<Routes>
|
|
1176
|
+
<Route path="/reservations" element={<ReservationPage />} />
|
|
1177
|
+
<Route path="/users" element={<UserManagementPage />} />
|
|
1178
|
+
</Routes>
|
|
1179
|
+
</Suspense>
|
|
1180
|
+
</Router>
|
|
1181
|
+
);
|
|
1182
|
+
}
|
|
1183
|
+
```
|
|
1184
|
+
|
|
1185
|
+
#### メモ化
|
|
1186
|
+
```tsx
|
|
1187
|
+
// ✅ 適切なメモ化
|
|
1188
|
+
const ExpensiveList = memo(function ExpensiveList({ items, filter }) {
|
|
1189
|
+
const filteredItems = useMemo(
|
|
1190
|
+
() => items.filter(item => item.type === filter),
|
|
1191
|
+
[items, filter]
|
|
1192
|
+
);
|
|
1193
|
+
|
|
1194
|
+
return (
|
|
1195
|
+
<ul>
|
|
1196
|
+
{filteredItems.map(item =>
|
|
1197
|
+
<ExpensiveListItem key={item.id} item={item} />
|
|
1198
|
+
)}
|
|
1199
|
+
</ul>
|
|
1200
|
+
);
|
|
1201
|
+
});
|
|
1202
|
+
```
|
|
1203
|
+
|
|
1204
|
+
## フロントエンド品質保証
|
|
1205
|
+
|
|
1206
|
+
### 1. **TypeScript活用**
|
|
1207
|
+
|
|
1208
|
+
```tsx
|
|
1209
|
+
// ✅ 型安全なAPI呼び出し
|
|
1210
|
+
interface CreateReservationRequest {
|
|
1211
|
+
roomId: string;
|
|
1212
|
+
startTime: string;
|
|
1213
|
+
endTime: string;
|
|
1214
|
+
purpose: string;
|
|
1215
|
+
}
|
|
1216
|
+
|
|
1217
|
+
interface ReservationResponse {
|
|
1218
|
+
id: string;
|
|
1219
|
+
status: 'confirmed' | 'pending' | 'cancelled';
|
|
1220
|
+
createdAt: string;
|
|
1221
|
+
}
|
|
1222
|
+
|
|
1223
|
+
async function createReservation(
|
|
1224
|
+
request: CreateReservationRequest
|
|
1225
|
+
): Promise<ReservationResponse> {
|
|
1226
|
+
const response = await fetch('/api/reservations', {
|
|
1227
|
+
method: 'POST',
|
|
1228
|
+
headers: { 'Content-Type': 'application/json' },
|
|
1229
|
+
body: JSON.stringify(request)
|
|
1230
|
+
});
|
|
1231
|
+
|
|
1232
|
+
if (!response.ok) {
|
|
1233
|
+
throw new Error('予約作成に失敗しました');
|
|
1234
|
+
}
|
|
1235
|
+
|
|
1236
|
+
return response.json();
|
|
1237
|
+
}
|
|
1238
|
+
```
|
|
1239
|
+
|
|
1240
|
+
### 2. **テスト戦略**
|
|
1241
|
+
|
|
1242
|
+
#### ユニットテスト
|
|
1243
|
+
```tsx
|
|
1244
|
+
// ✅ Custom Hooksのテスト
|
|
1245
|
+
import { renderHook, act } from '@testing-library/react';
|
|
1246
|
+
import { useReservation } from './useReservation';
|
|
1247
|
+
|
|
1248
|
+
test('should create reservation successfully', async () => {
|
|
1249
|
+
const { result } = renderHook(() => useReservation());
|
|
1250
|
+
|
|
1251
|
+
await act(async () => {
|
|
1252
|
+
await result.current.createReservation({
|
|
1253
|
+
roomId: '1',
|
|
1254
|
+
startTime: '2024-01-01T10:00:00Z',
|
|
1255
|
+
endTime: '2024-01-01T11:00:00Z',
|
|
1256
|
+
purpose: 'ミーティング'
|
|
1257
|
+
});
|
|
1258
|
+
});
|
|
1259
|
+
|
|
1260
|
+
expect(result.current.reservations).toHaveLength(1);
|
|
1261
|
+
});
|
|
1262
|
+
```
|
|
1263
|
+
|
|
1264
|
+
#### 統合テスト
|
|
1265
|
+
```tsx
|
|
1266
|
+
// ✅ コンポーネント統合テスト
|
|
1267
|
+
import { render, screen, fireEvent, waitFor } from '@testing-library/react';
|
|
1268
|
+
import { ReservationForm } from './ReservationForm';
|
|
1269
|
+
|
|
1270
|
+
test('should submit reservation form', async () => {
|
|
1271
|
+
const mockOnSubmit = jest.fn();
|
|
1272
|
+
|
|
1273
|
+
render(<ReservationForm onSubmit={mockOnSubmit} />);
|
|
1274
|
+
|
|
1275
|
+
fireEvent.change(screen.getByLabelText('会議室'), {
|
|
1276
|
+
target: { value: '会議室A' }
|
|
1277
|
+
});
|
|
1278
|
+
fireEvent.click(screen.getByText('予約する'));
|
|
1279
|
+
|
|
1280
|
+
await waitFor(() => {
|
|
1281
|
+
expect(mockOnSubmit).toHaveBeenCalledWith(
|
|
1282
|
+
expect.objectContaining({
|
|
1283
|
+
roomName: '会議室A'
|
|
1284
|
+
})
|
|
1285
|
+
);
|
|
1286
|
+
});
|
|
1287
|
+
});
|
|
1288
|
+
```
|
|
1289
|
+
|
|
1290
|
+
## フロントエンド移行戦略
|
|
1291
|
+
|
|
1292
|
+
### 段階的移行アプローチ
|
|
1293
|
+
|
|
1294
|
+
```plantuml
|
|
1295
|
+
@startuml
|
|
1296
|
+
start
|
|
1297
|
+
|
|
1298
|
+
:現行フロントエンド分析;
|
|
1299
|
+
|
|
1300
|
+
if (移行戦略) then (段階的)
|
|
1301
|
+
:新機能を新アーキテクチャで実装;
|
|
1302
|
+
:既存機能の段階的リファクタリング;
|
|
1303
|
+
else (リライト)
|
|
1304
|
+
:新システムを並行開発;
|
|
1305
|
+
:フィーチャーフラグによる切り替え;
|
|
1306
|
+
endif
|
|
1307
|
+
|
|
1308
|
+
:共通ライブラリの整備;
|
|
1309
|
+
:コンポーネントライブラリの移行;
|
|
1310
|
+
:状態管理の統合;
|
|
1311
|
+
|
|
1312
|
+
:E2Eテストによる動作確認;
|
|
1313
|
+
:パフォーマンステスト;
|
|
1314
|
+
|
|
1315
|
+
:本番環境移行;
|
|
1316
|
+
:監視・メトリクス設定;
|
|
1317
|
+
|
|
1318
|
+
stop
|
|
1319
|
+
@enduml
|
|
1320
|
+
```
|
|
1321
|
+
|
|
1322
|
+
## アーキテクチャ移行戦略
|
|
1323
|
+
|
|
1324
|
+
### 段階的移行アプローチ
|
|
1325
|
+
|
|
1326
|
+
```plantuml
|
|
1327
|
+
@startuml
|
|
1328
|
+
start
|
|
1329
|
+
|
|
1330
|
+
:現状アーキテクチャ分析;
|
|
1331
|
+
:移行計画策定;
|
|
1332
|
+
|
|
1333
|
+
fork
|
|
1334
|
+
:既存機能の段階的抽出;
|
|
1335
|
+
fork again
|
|
1336
|
+
:新機能の新アーキテクチャ実装;
|
|
1337
|
+
end fork
|
|
1338
|
+
|
|
1339
|
+
:境界コンテキスト定義;
|
|
1340
|
+
:データ移行計画;
|
|
1341
|
+
:テスト戦略見直し;
|
|
1342
|
+
|
|
1343
|
+
fork
|
|
1344
|
+
:モジュール単位での移行;
|
|
1345
|
+
fork again
|
|
1346
|
+
:継続的統合の維持;
|
|
1347
|
+
end fork
|
|
1348
|
+
|
|
1349
|
+
:移行完了確認;
|
|
1350
|
+
:旧システム廃棄;
|
|
1351
|
+
|
|
1352
|
+
stop
|
|
1353
|
+
@enduml
|
|
1354
|
+
```
|
|
1355
|
+
|
|
1356
|
+
### 移行時の注意点
|
|
1357
|
+
|
|
1358
|
+
#### バックエンド移行
|
|
1359
|
+
1. **段階的移行**: ビッグバン移行は避け、モジュール単位で実施
|
|
1360
|
+
2. **後方互換性**: 既存のAPIやインターフェースの維持
|
|
1361
|
+
3. **データ整合性**: 移行期間中のデータ同期戦略
|
|
1362
|
+
4. **テストカバレッジ**: 移行前後でのテスト品質維持
|
|
1363
|
+
5. **パフォーマンス監視**: 移行による性能劣化の早期検知
|
|
1364
|
+
|
|
1365
|
+
#### フロントエンド移行
|
|
1366
|
+
1. **フィーチャーフラグ**: 新旧機能の段階的切り替え
|
|
1367
|
+
2. **コンポーネント移行**: デザインシステムの統一
|
|
1368
|
+
3. **状態管理移行**: データフローの一貫性維持
|
|
1369
|
+
4. **バンドルサイズ監視**: パフォーマンス劣化防止
|
|
1370
|
+
5. **ブラウザ互換性**: サポート対象ブラウザでの動作確認
|
|
1371
|
+
|
|
1372
|
+
## まとめ
|
|
1373
|
+
|
|
1374
|
+
### バックエンドアーキテクチャ選択の指針
|
|
1375
|
+
|
|
1376
|
+
1. **業務ドメインの理解**: 中核業務と補完業務の明確な識別
|
|
1377
|
+
2. **データ複雑性の評価**: エンティティ関係とビジネスルールの複雑さ
|
|
1378
|
+
3. **非機能要件の考慮**: パフォーマンス、スケーラビリティ、監査要件
|
|
1379
|
+
4. **チーム能力の評価**: 技術的習熟度と学習意欲
|
|
1380
|
+
5. **長期戦略の整合**: 将来的な拡張性と保守性の確保
|
|
1381
|
+
|
|
1382
|
+
### フロントエンドアーキテクチャ選択の指針
|
|
1383
|
+
|
|
1384
|
+
1. **プロジェクト規模**: 小規模・中規模・大規模に応じた構造設計
|
|
1385
|
+
2. **ユーザー体験要件**: SEO・パフォーマンス・インタラクティビティの優先順位
|
|
1386
|
+
3. **状態管理複雑性**: データフロー・更新頻度・共有範囲の分析
|
|
1387
|
+
4. **チーム構成**: フロントエンド専任・フルスタック・QAエンジニアの有無
|
|
1388
|
+
5. **技術的制約**: ブラウザサポート・既存システム・運用環境
|
|
1389
|
+
|
|
1390
|
+
### 成功要因
|
|
1391
|
+
|
|
1392
|
+
#### バックエンド
|
|
1393
|
+
- **適切なパターン選択**: 業務特性に応じた最適なアーキテクチャの採用
|
|
1394
|
+
- **段階的な実装**: 小さく始めて継続的に改善
|
|
1395
|
+
- **品質の作り込み**: テストファースト・継続的インテグレーション
|
|
1396
|
+
- **ドメイン駆動設計**: ビジネスルールの適切な表現と分離
|
|
1397
|
+
|
|
1398
|
+
#### フロントエンド
|
|
1399
|
+
- **適切な技術選択**: 要件に応じたライブラリ・フレームワークの選定
|
|
1400
|
+
- **コンポーネント設計**: 再利用性と保守性を両立した設計
|
|
1401
|
+
- **状態管理戦略**: スケールに応じた適切な状態管理手法の採用
|
|
1402
|
+
- **パフォーマンス最適化**: ユーザー体験を重視したパフォーマンス設計
|
|
1403
|
+
|
|
1404
|
+
### 統合的成功要因
|
|
1405
|
+
|
|
1406
|
+
- **チーム教育**: アーキテクチャ原則の共有と実践
|
|
1407
|
+
- **継続的改善**: 定期的な設計見直しとリファクタリング
|
|
1408
|
+
- **品質文化**: 自動テスト・コードレビュー・静的解析の徹底
|
|
1409
|
+
- **段階的移行**: リスクを最小化した漸進的なアーキテクチャ改善
|
|
1410
|
+
|
|
1411
|
+
### 実践ポイント
|
|
1412
|
+
|
|
1413
|
+
#### 開発初期
|
|
1414
|
+
1. **要件に基づく技術選択**: 過度な技術より要件適合性を重視
|
|
1415
|
+
2. **プロトタイプ開発**: アーキテクチャ妥当性の早期検証
|
|
1416
|
+
3. **チーム合意形成**: 技術選択の背景と理由の共有
|
|
1417
|
+
|
|
1418
|
+
#### 開発中期
|
|
1419
|
+
1. **継続的リファクタリング**: 設計負債の早期解消
|
|
1420
|
+
2. **パフォーマンス監視**: 品質メトリクスに基づく改善
|
|
1421
|
+
3. **テストカバレッジ**: アーキテクチャ変更に対する安全性確保
|
|
1422
|
+
|
|
1423
|
+
#### 運用期
|
|
1424
|
+
1. **監視・メトリクス**: 実装の効果測定と改善指針
|
|
1425
|
+
2. **技術的負債管理**: 計画的なアーキテクチャ改善
|
|
1426
|
+
3. **スケーラビリティ対応**: 成長に応じたアーキテクチャ進化
|
|
1427
|
+
|
|
1428
|
+
良いアーキテクチャは、システムの長期的成功の基盤となります。本ガイドを参考に、バックエンド・フロントエンドそれぞれの特性を理解し、プロジェクトに最適なアーキテクチャを選択・実装してください。
|