ページの作成
親となるページを選択してください。
親ページに紐づくページを子ページといいます。
例: 親=スポーツ, 子1=サッカー, 子2=野球
子ページを親ページとして更に子ページを作成することも可能です。
例: 親=サッカー, 子=サッカーのルール
親ページはいつでも変更することが可能なのでとりあえず作ってみましょう!
| この記事の要点 |
|---|
|
@Service の基本
@Service
public class UserService {
private final UserRepository userRepository;
public UserService(UserRepository userRepository) {
this.userRepository = userRepository;
}
public User findById(Long id) {
return userRepository.findById(id)
.orElseThrow(() -> new EntityNotFoundException());
}
@Transactional
public User createUser(String name, String email) {
User user = new User(name, email);
return userRepository.save(user);
}
}
@Service vs @Component の違い
機能的には同じ(どちらも Bean 登録)。違いは意味付けだけ:
| 項目 | @Service | @Component |
|---|---|---|
| Bean 登録 | ○ | ○ |
| 意味付け | 「ビジネスロジック層」 | 「汎用 Bean」 |
| AOP ターゲット | ポイントカット @within(...@Service) で対象に | 普通の対象 |
| 使う場面 | サービス層全般 | ヘルパー・ユーティリティ |
結論: ビジネスロジックを書くなら @Service を選ぶ(コード読む人にも意図が伝わる)。
典型的なレイヤ構成
// 1. Controller 層 (Web)
@RestController
@RequestMapping("/api/users")
public class UserController {
private final UserService userService;
public UserController(UserService userService) {
this.userService = userService;
}
@PostMapping
public UserDto create(@RequestBody @Valid UserCreateRequest req) {
return UserDto.from(userService.createUser(req.getName(), req.getEmail()));
}
}
// 2. Service 層 (ビジネスロジック)
@Service
public class UserService {
private final UserRepository userRepository;
private final NotificationService notificationService;
public UserService(UserRepository ur, NotificationService ns) {
this.userRepository = ur;
this.notificationService = ns;
}
@Transactional
public User createUser(String name, String email) {
// バリデーション
if (userRepository.existsByEmail(email)) {
throw new BusinessException("メールアドレス重複");
}
// 保存
User user = userRepository.save(new User(name, email));
// 通知 (別サービス呼び出し)
notificationService.sendWelcomeEmail(user);
return user;
}
}
// 3. Repository 層 (DB アクセス)
@Repository
public interface UserRepository extends JpaRepository {
Optional findByEmail(String email);
boolean existsByEmail(String email);
}
@Service と @Transactional
サービス層に @Transactional を付けるのが定石:
@Service
@Transactional(readOnly = true) // クラスレベルは読み取り専用 (推奨)
public class UserService {
public User findById(Long id) {
// readOnly = true (デフォルト)
return userRepository.findById(id).orElseThrow();
}
@Transactional // 書き込み系はメソッドで上書き
public User updateUser(Long id, String name) {
User user = findById(id);
user.setName(name);
return user; // dirty checking で UPDATE
}
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void logActivity(Long userId, String action) {
// 親トランザクションと別に独立トランザクションで実行
}
@Transactional(rollbackFor = Exception.class) // checked exception でも rollback
public void riskyOperation() throws IOException {
// ...
}
}
@Service のテスト
// 単体テスト (Mockito)
@ExtendWith(MockitoExtension.class)
class UserServiceTest {
@Mock
private UserRepository userRepository;
@InjectMocks
private UserService userService;
@Test
void createUser_success() {
// given
User saved = new User("Alice", "alice@test.com");
saved.setId(1L);
when(userRepository.existsByEmail("alice@test.com")).thenReturn(false);
when(userRepository.save(any(User.class))).thenReturn(saved);
// when
User result = userService.createUser("Alice", "alice@test.com");
// then
assertEquals(1L, result.getId());
verify(userRepository).save(any(User.class));
}
@Test
void createUser_duplicateEmail() {
when(userRepository.existsByEmail("dup@test.com")).thenReturn(true);
assertThrows(BusinessException.class, () ->
userService.createUser("Bob", "dup@test.com")
);
}
}
インタフェース + 実装パターン
// インタフェース
public interface UserService {
User findById(Long id);
User createUser(String name, String email);
}
// 実装
@Service
public class UserServiceImpl implements UserService {
private final UserRepository userRepository;
public UserServiceImpl(UserRepository ur) {
this.userRepository = ur;
}
@Override
public User findById(Long id) {
return userRepository.findById(id).orElseThrow();
}
@Override
public User createUser(String name, String email) {
// ...
}
}
// メリット:
// - テストでモック化しやすい
// - 実装差し替え可能 (本番 / モック)
// - DI でインタフェース型で受け取れる
// 注: Spring Boot 2+ ではプロキシ生成のため
// インタフェース不要でも動く (CGLIB proxy)
// シンプルなら実装クラスだけで OK
命名規則
- サービス名:
UserService/OrderService/EmailService - 実装クラス:
UserServiceImpl(インタフェース有り) または単にUserService - メソッド名:
find/create/update/delete(ビジネス操作の動詞) - パッケージ:
com.example.service等で分離
関連記事
ページの作成
親となるページを選択してください。
親ページに紐づくページを子ページといいます。
例: 親=スポーツ, 子1=サッカー, 子2=野球
子ページを親ページとして更に子ページを作成することも可能です。
例: 親=サッカー, 子=サッカーのルール
親ページはいつでも変更することが可能なのでとりあえず作ってみましょう!
子ページ
子ページはありません
人気ページ
- 1 Eclipseで「サーバーに追加または除去できるリソースがありません。」の原因と対処法
- 2 tomcat の起動 / 停止ログと catalina.log・catalina.out の違い
- 3 JavaScript base URL 取得方法|window.location.origin と SSR/Node.js 対応
- 4 YouTube Data API v3 エラー一覧|403/400/404 の主要原因と切り分け
- 5 Spring Frameworkのアノテーション一覧
- 6 Laravel エラー一覧|500/Blade/DB 接続/ルーティングの代表エラー
- 7 3Dグラフィックスとは|モデリング/レンダリング/主要ソフトウェア (Blender / Maya)
- 8 【Spring】@Valueアノテーションとは
- 9 CATALINA_HOME の確認方法 (Linux / Mac)
- 10 【Spring】@Autowiredアノテーションとは
最近更新/作成されたページ
- Laravel キャッシュクリア完全ガイド(cache:clear / config:clear / 2026-05-18 07:42:07
- プロジェクトの作成と削除 2026-05-18 07:42:07
- インストール直後にNetbeansが反応しない 2026-05-18 07:42:07
- 動画やチャンネルの検索 2026-05-18 07:42:07
- APIキー取得方法 2026-05-18 07:42:07
- チャンネル情報の取得 2026-05-18 07:42:07
- API 入門 — Web API(REST / GraphQL / gRPC / 2026-05-18 07:42:07
- インストール(eclipseプラグイン) 2026-05-18 07:42:07
- Laravel「Dotenv values containing spaces must be surrounded 2026-05-18 07:42:07
- エラー一覧 2026-05-18 07:42:07
- curl: (51) SSL: certificate subject name '~' does not match 2026-05-18 07:42:07
- インストール方法(Windows版) 2026-05-18 07:42:07
- JSONから配列に変換 2026-05-18 07:42:07
- 処理を一定時間待つ 2026-05-18 07:42:07
- A non well formed numeric value encountered 2026-05-18 07:42:07
コメントを削除してもよろしいでしょうか?