728x90

1. 개념

1.1 DB Lock 이란?

  • DB Lock은 여러 트랜잭션이 동시에 같은 자원에 접근할 때 데이터 무결성(정합성)을 보장하기 위해 사용되는 메커니즘
  • 쉽게 말해, 어떤 사용자가 데이터를 사용하고 있는 동안에는 다른 사용자가 그 데이터를 동시에 수정하면 안 되기 때문에, ‘잠금’을 걸어서 충돌(Conflict)을 방지

1.2 왜 Lock이 필요한가?

  • 여러 사용자가 동시에 DB를 사용하는 경우 같은 데이터를 동시에 수정하려고 시도하는 경우 충돌 제어 및 데이터의 일관성을 보장하기 위해

1.3 언제 Lock을 사용하는가?

  • 사용해야 하는 경우!
    • 동시에 읽기/쓰기가 빈번하게 일어나는 중요한 테이블에 대해, 데이터 무결성을 엄격히 보장해야 할 때.
    • 은행 이체, 재고 관리, 주문 처리와 같이 동시에 발생하면 안 되는 시나리오를 제어할 때.
  • 주의해야할 점!
    • 불필요한 락은 성능 저하를 유발할 수 있으므로, 적절한 수준에서 사용하는 것이 중요
    • DeadLock!!
      • DeadLock발생 여부를 확인해야함!

1.4 Lock의 생명주기

1.4.1 트랜잭션 시작!

  • DB와의 트랜잭션을 시작
  • 아직 특정 자원에 대한 락이 획득된 상태는 🙅

1.4.2 Lock 획득

  1. 쿼리 요청
  2. Lock 체크
  3. 대기 혹은 Lock 획득

1.4.3 트랜잭션 종료

  1. 트랜잭션 커밋 혹은 롤백
  2. 락 해제

2. Lock 종류

2.1 낙관적 락 (충돌이 발생하지 않을 것이라고 가정)

  • 데이터베이스의 락을 사용하는 것이 아닌 application 레벨에서 버전 관리
  • 특정 작업을 수행하기 전에 별도의 락(lock)을 걸지 않고, 작업 완료 시점에서 데이터의 변경 여부를 확인하여 충돌 여부를 판단 (@Version 활용)
  • 데이터 충돌이 거의 없을것이라고 가정한 경우 사용
    • LockModeType.OPTIMISTIC로 적용
  • 충돌 시 ObjectOptimisticLockingFailureException 발생

2.2 비관적 락 (충돌이 자주 발생할 것이라고 가정)

2.2.1 공유락(Shared Lock, S Lock)

  • 여러 트랜잭션이 동시에 데이터를 읽기할 수 있지만, 쓰기 하려면 공유락을 해제하고 배타락으로 변경
  • 다른 트랜잭션이 쓰기하려 하면 대기 상태
    • LockModeType.PESSIMISTIC_READ로 적용

2.2.2 배타락(Exclusive Lock, X Lock)

  • 오직 한 트랜잭션만 해당 데이터를 읽거나 쓸 수 있음
  • 다른 트랜잭션이 접근하려 하면 대기 상태
    • LockModeType.PESSIMISTIC_WRITE로 적용

2.3 분산 락

  • 여러 인스턴스나 분산 환경에서 락을 설정
  • 데이터베이스에 직접 Lock을 걸지 않고, 외부에서 권한을 받아 처리
  • Redis, **Zookeeper 등... 을 활용하여 적용**

 

반응형

'Study > Spring' 카테고리의 다른 글

[Spring] Spring Batch  (0) 2025.01.23
[Spring] Spring Security + OAuth 2.0 소셜 로그인  (1) 2025.01.16
[Spring] QueryDSL-JPA  (0) 2024.12.30
[Spring] @InitBinder  (0) 2021.08.08
[Spring] @Data 어노테이션  (0) 2021.06.30
728x90

1. QueryDSL(Query Domain-Specific Language)

1.1 💡QueryDSL 이란?

  • SQL, JPQL 같은 쿼리를 Java 코드로 작성할 수 있게 해주는 타입 안전한 ORM 기반 쿼리 빌더
  • SQLJava 코드로 작성하면서, 가독성과 유지보수성을 대폭 개선할 수 있는 도구

1.2 ☝️ 왜 사용해야할까

1.2.1 SQL/JPQL의 한계 해결

  • 문법 오류 - 쿼리의 오류를 런타임에서만 확인 가능
  • 필드 이름 변경 - 엔티티 필드가 변경되면 문자열 기반 쿼리에서 이를 수동으로 수정해야 함.
  • 동적 쿼리 작성의 어려움 - 조건에 따라 쿼리를 유연하게 작성하려면 코드 복잡도 향상
  • 코드 비교
// JPQL
// 필터가 많아질수록 코드가 복잡
public List<User> findUsersDynamicJPQL(EntityManager em, Integer age, String name) {
    StringBuilder jpql = new StringBuilder("SELECT u FROM User u WHERE 1=1");

    if (age != null) {
        jpql.append(" AND u.age > :age");
    }
    if (name != null) {
        jpql.append(" AND u.name = :name");
    }

    TypedQuery<User> query = em.createQuery(jpql.toString(), User.class);
    if (age != null) query.setParameter("age", age);
    if (name != null) query.setParameter("name", name);

    return query.getResultList();
}


// QueryDSL
// BooleanBuilder 사용
public List<User> findUsersDynamicQueryDSL(JPAQueryFactory queryFactory, Integer age, String name) {
    QUser user = QUser.user;

    BooleanBuilder builder = new BooleanBuilder(); // 동적 조건 빌더
    if (age != null) {
        builder.and(user.age.gt(age));
    }
    if (name != null) {
        builder.and(user.name.eq(name));
    }

    return queryFactory
            .selectFrom(user)
            .where(builder) // 동적 조건 적용
            .fetch();
}

1.2.2 복잡한 쿼리 요구사항 처리

  • 조인 - 여러 테이블 데이터를 조합해서 처리.
  • 서브쿼리 - 쿼리 안에 또 다른 쿼리를 포함.
  • 그룹핑 및 집계 - 데이터를 그룹화하거나 통계 처리.

1.2.3 장/단점

  • 장점
    • 타입 안전한 쿼리 작성
    • 가독성 UP, 유지보수성 UP
    • 동적 쿼리 작성의 유연함
    • 다양한 쿼리 지원
  • 단점
    • 초기 세팅의 복잡성, 의존성 관리
    • Q 클래스 생성 필요
    • 러닝 커브

1.3 💁 언제 사용해야할까?

1.3.1 🙆 사용해야하는 경우

  • 동적 쿼리가 많은 프로젝트
  • 복잡한 쿼리를 자주 작성해야 하는 대규모 애플리케이션
  • 타입 안전성과 코드 가독성을 중시하는 팀 환경

1.3.2 🙅 사용하지 말아야 하는 경우

  • 프로젝트가 간단한 CRUD 중심인 경우
  • 네이티브 SQL을 주로 사용하는 환경

1.4 ✈️ 대안

1.4.1 JPA Criteria API

  • JPA 표준 스펙에 포함된 쿼리 빌더 API

✅ 장점

  • 표준 API: JPA 구현체(Hibernate 등)와 상관없이 동작.
  • 별도의 외부 라이브러리 없이 사용 가능.

❌ 단점

  • 가독성 부족: 코드가 복잡하고 장황해지기 쉬움.
  • 유지보수가 어려움.
  • 예제 코드
CriteriaBuilder cb = em.getCriteriaBuilder();
CriteriaQuery<User> query = cb.createQuery(User.class);
Root<User> user = query.from(User.class);

Predicate agePredicate = cb.gt(user.get("age"), 18);
Predicate statusPredicate = cb.equal(user.get("status"), "ACTIVE");

query.select(user).where(cb.and(agePredicate, statusPredicate));
List<User> result = em.createQuery(query).getResultList();

1.4.2 Spring Data JPA Specification

  • Spring Data JPA에서 제공하는 Specification 인터페이스를 활용하면 동적 쿼리 구현

✅ 장점

  • Spring Data JPA와 통합: Spring 프로젝트와 자연스럽게 연동.
  • 동적 조건 작성이 비교적 간단.

❌ 단점

  • QueryDSL에 비해 표현력이 제한적.
  • 복잡한 쿼리 작성에는 적합하지 않음.
  • 예제코드
public class UserSpecifications {
    public static Specification<User> hasAgeGreaterThan(int age) {
        return (root, query, cb) -> cb.gt(root.get("age"), age);
    }

    public static Specification<User> hasStatus(String status) {
        return (root, query, cb) -> cb.equal(root.get("status"), status);
    }
}

// 사용
Specification<User> spec = Specification.where(UserSpecifications.hasAgeGreaterThan(18))
                                         .and(UserSpecifications.hasStatus("ACTIVE"));

List<User> users = userRepository.findAll(spec);

1.4.3 네이티브 SQL

✅ 장점

  • 유연성: 데이터베이스에 특화된 쿼리를 작성 가능
  • 성능 최적화에 유리

❌ 단점

  • 타입 안전성 부족: 문자열 기반으로 작성
  • 데이터베이스 종속적 → 이식성이 낮음
  • 유지보수가 어려움
  • 예제 코드
@Query(value = "SELECT * FROM users WHERE age > :age AND status = :status", nativeQuery = true)
List<User> findUsersNative(@Param("age") int age, @Param("status") String status);

1.4.4 jOOQ (Java Object Oriented Querying)

  • jOOQ는 SQL을 Java 코드로 작성하면서도, 데이터베이스와 밀접하게 통합하여 고급 기능을 사용할 수 있는 강력한 도구

✅ 장점

  • SQL 친화적 설계: SQL의 고급 기능(윈도우 함수, CTE 등)을 Java 코드로 직접 활용 가능.
  • 타입 안전성: SQL 컬럼과 Java 필드 간의 타입을 컴파일 타임에 검증.
  • DBMS 특화 기능 지원: 데이터베이스에 따라 최적화된 SQL 생성 가능.
  • 코드 자동 생성: 테이블 스키마 기반으로 Java 코드를 생성하여, 변경 사항을 자동 반영.

❌ 단점

  • SQL 종속성: 데이터베이스 변경 시 쿼리 코드 수정이 필요.
  • 학습 난이도: SQL에 익숙하지 않은 개발자에게는 진입 장벽이 높음.
  • ORM 부족: 객체 지향적 추상화가 부족하고, SQL 중심의 설계에 초점.
  • 예제 코드
// 동적 조건 추가
public List<User> findUsersDynamicJooq(DSLContext context, Integer age, String name) {
    Condition condition = DSL.trueCondition();

    if (age != null) {
        condition = condition.and(USER.AGE.gt(age));
    }
    if (name != null) {
        condition = condition.and(USER.NAME.eq(name));
    }

    return context.selectFrom(USER)
                  .where(condition)
                  .fetchInto(User.class);
}

 

 

라이브러리 특징 장점 단점
QueryDSL - 타입 안전한 동적 쿼리 작성을 지원
- Q클래스를 통해 메서드 체인 방식으로 쿼리를 작성
- 타입 안전성(Type Safety) 보장
- 가독성 높은 코드 작성 가능
- IDE 자동완성 지원
- 초기 설정 복잡
- Q클래스 생성 필요
Spring Data JPA Specifications - JPA의 Criteria API를 기반으로 동적 쿼리를 작성
- Specification 인터페이스를 사용
- 추가 라이브러리 없이 Spring Data JPA와 자연스러운 통합
- 동적 쿼리 작성 가능
- 코드가 장황하고 가독성이 떨어질 수 있음
JOOQ - SQL 빌더 라이브러리로, SQL 문법을 코드로 표현
- 고급 SQL 기능을 지원
- SQL 문법을 코드로 직관적으로 표현 가능
- 복잡한 쿼리와 고급 SQL 기능 활용 가능
- JPA와 통합하려면 추가 설정 필요
Criteria API (JPA 기본) - JPA 표준 사양
- 메서드 체인 방식을 통해 동적 쿼리를 작성
- 모든 JPA 구현체에서 사용 가능
- 추가 의존성 필요 없음
- 코드가 복잡하고 가독성이 떨어짐
Native QSL - SQL 문법그대로 쿼리 작성 - 데이터베이스에 특화된 쿼리를 작성 가능
- 성능 최적화에 유리
- 데이터베이스 종속적 → 이식성이 낮음
- 유지보수가 어려움

 

반응형

'Study > Spring' 카테고리의 다른 글

[Spring] Spring Security + OAuth 2.0 소셜 로그인  (1) 2025.01.16
[Spring Boot] DB Lock  (0) 2025.01.10
[Spring] @InitBinder  (0) 2021.08.08
[Spring] @Data 어노테이션  (0) 2021.06.30
[Spring] 영속성 컨텍스트 (Persistence Context)  (0) 2021.06.30
728x90

💡 TDD(Test-Driven Development): 실패에서 시작하는 개발의 예술

1. TDD(Test-Driven Development)

🎯1.1 TDD란 무엇인가요?

  • TDD는 개발자의 ‘습관’
    • “테스트부터 작성하자. 그리고 통과시키자!”
  • 정의: 테스트를 먼저 작성하고, 그 테스트를 통과하기 위해 코드를 작성하는 개발 방법론.
  • 왜 중요한가요?
    • 실패를 두려워하지 않고, 더 나은 코드를 만들어 가는 과정!
    • Red-Green-Refactor라는 단순하지만 강력한 루틴으로 코드 품질을 높임.

🚦 1.2 기본 사이클: Red-Green-Refactor

1.2.1 Red: 실패를 즐기자!

  • 테스트부터 작성 (아직 구현은 X).
  • 실패를 통해 요구사항과 문제를 명확히 이해.
import org.junit.jupiter.api.Test;
import static org.junit.jupiter.api.Assertions.assertEquals;

public class CalculatorTest {

    @Test
    void testAddition() {
        Calculator calculator = new Calculator();
        int result = calculator.add(2, 3);
        assertEquals(5, result); // 실패! 왜? 구현 코드가 없으니까.
    }
}

1.2.2 Green: 테스트 통과는 최소 노력으로.

  • 테스트를 통과시키는 코드 작성.
  • 복잡하게 고민하지 말고 “빨리 테스트 통과”에 집중!
public class Calculator {
    public int add(int a, int b) {
        return a + b; // 통과! 왜? 테스트가 요구한 기능만 구현했으니까.
    }
}

1.2.3 Refactor: 이제 진짜 멋지게!

  • 읽기 좋은 코드, 유지보수하기 쉬운 코드로 개선.
  • 코드의 동작은 그대로, 구조는 깔끔하게!
public class Calculator {
    // 향후 확장을 고려해 더 나은 구조로 개선
    public int add(int a, int b) {
        return calculate(a, b, "+");
    }

    // 공통 로직 처리 메서드 추가
    private int calculate(int a, int b, String operation) {
        switch (operation) {
            case "+":
                return a + b;
            default:
                throw new UnsupportedOperationException("Operation not supported");
        }
    }
}

✨ 1.3 왜 중요한가요?

  • 🛡 안정성: 모든 기능에 대한 자동화된 테스트 보장.
  • 📈 생산성 향상: 리팩토링 부담 감소.

“TDD를 하면 시간 낭비라고?” 아니요! 미래의 당신이 감사할 겁니다. 😎

🔥 1.4 장단점

  • 장점
    • 코드의 신뢰성 증가.
    • 요구사항 변경에도 흔들리지 않는 튼튼한 구조.
    • 버그 감소로 디버깅 시간 절약.
  • 단점
    • 초기 진입장벽 존재.
    • 테스트 작성에 시간 투자 필요.
    • 잘못된 테스트 설계 시 오히려 발목을 잡을 수 있음.

🙀 1.5 DDD(Domain Driven Design)와의 관계

  • TDD는 테스트를 통해 코드를 어떻게 구현할지에 초점을 둔 개발 방법론
  • DDD는 도메인 모델을 설계하여 무엇을 구현할지를 정의하는 철학
질문 TDD DDD
왜 사용하는가? 기능의 동작을 보장하기 위해 비즈니스 요구사항을 시스템에 반영하기 위해
왜 시작하는가? 코드를 테스트하면서 명확한 피드백을 얻기 위해 요구사항과 도메인 모델을 기반으로 설계하기 위해
 중요한가? 코드의 안정성과 품질을 높이기 위해 복잡한 비즈니스 로직을 제대로 반영하기 위해
어떻게 다른가? 테스트 작성 → 코드 작성 → 리팩토링 요구사항 분석 → 도메인 설계 → 코드 작성

2. 테스트 코드

💡2.1 테스트 코드란 무엇인가요?

  • “내 코드, 정말 제대로 작동할까?”라는 질문에 대한 답을 미리 준비하는 개발자의 무기
  • 프로그램의 기능이 의도대로 동작하는지 자동으로 확인

🛠 2.2 종류

2.2.1 단위 테스트 (Unit Test)

  • 특징: 순수 비즈니스 로직(단일 메서드나 클래스)을 테스트
  • 도구: @Mock, Mockito.
  • 예시: UserService가 UserRepository를 올바르게 호출하는지 테스트
import org.junit.jupiter.api.Test;
import org.mockito.Mock;
import org.mockito.MockitoAnnotations;

import java.util.Optional;

import static org.mockito.Mockito.*;
import static org.junit.jupiter.api.Assertions.*;

class UserServiceTest {

    @Mock
    private UserRepository userRepository;

    private UserService userService;

    @Test
    void should_ReturnUser_When_ValidIdProvided() {
        // Mock 초기화
        MockitoAnnotations.openMocks(this);
        userService = new UserService(userRepository);

        // Given
        User mockUser = new User(1L, "Alice");
        when(userRepository.findById(1L)).thenReturn(Optional.of(mockUser));

        // When
        User result = userService.getUserById(1L);

        // Then
        assertEquals("Alice", result.getName());
        verify(userRepository, times(1)).findById(1L);
    }
}

2.2.2 통합 테스트 (Integration Test)

  • 특징: 여러 계층(Service, Repository 등)이 올바르게 연동되는지 확인
  • 도구: @SpringBootTest
  • 예시: 데이터 저장 및 조회가 잘 동작하는지 확인.
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.transaction.annotation.Transactional;

import static org.junit.jupiter.api.Assertions.*;

@SpringBootTest
@Transactional // 테스트 후 DB 롤백
class UserIntegrationTest {

    @Autowired
    private UserService userService;

    @Autowired
    private UserRepository userRepository;

    @Test
    void should_SaveAndRetrieveUser() {
        // Given
        User newUser = new User(null, "Bob");

        // When
        userService.saveUser(newUser);
        User retrievedUser = userService.getUserById(newUser.getId());

        // Then
        assertNotNull(retrievedUser);
        assertEquals("Bob", retrievedUser.getName());
    }
}

2.2.3 End-to-End 테스트 (E2E Test)

  • 특징: 시스템의 전체 흐름이 사용자 관점에서 잘 작동하는지 검증
  • 도구: @SpringBootTest, TestRestTemplate
  • 예시: 사용자 생성 후 조회 API가 올바르게 동작하는지 확인.
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.boot.test.web.client.TestRestTemplate;
import org.springframework.boot.web.server.LocalServerPort;

import static org.junit.jupiter.api.Assertions.*;

@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
class UserE2ETest {

    @LocalServerPort
    private int port;

    @Autowired
    private TestRestTemplate restTemplate;

    @Test
    void should_CreateAndRetrieveUser() {
        // Given
        String baseUrl = "http://localhost:" + port + "/users";
        UserRequest userRequest = new UserRequest("Charlie");

        // Step 1: 사용자 생성
        UserResponse createdUser = restTemplate.postForObject(baseUrl, userRequest, UserResponse.class);

        assertNotNull(createdUser);
        assertEquals("Charlie", createdUser.getName());

        // Step 2: 생성된 사용자 조회
        UserResponse retrievedUser = restTemplate.getForObject(baseUrl + "/" + createdUser.getId(), UserResponse.class);

        assertNotNull(retrievedUser);
        assertEquals("Charlie", retrievedUser.getName());
    }
}

🚦 2.3 Spring Boot 테스트에서 사용하는 Mock의 종류

2.3.1 @Mock

  • @Mock은 객체의 동작을 정의해서 의존성을 가짜로 만들어 줍니다.
  • 주로 단위 테스트에서 사용하며, Mock 라이브러리인 Mockito를 함께 사용합니다.
import org.junit.jupiter.api.Test;
import org.mockito.Mock;
import org.mockito.MockitoAnnotations;
import static org.mockito.Mockito.*;
import static org.junit.jupiter.api.Assertions.*;

class UserServiceTest {

    @Mock
    private UserRepository userRepository; // 독립된 Mock 객체 생성

    private UserService userService;

    @Test
    void should_ReturnUser_When_ValidIdProvided() {
        // Mock 초기화
        MockitoAnnotations.openMocks(this);
        userService = new UserService(userRepository);

        // Given
        when(userRepository.findById(1L)).thenReturn(Optional.of(new User(1L, "Alice")));

        // When
        User result = userService.getUserById(1L);

        // Then
        assertEquals("Alice", result.getName());
        verify(userRepository, times(1)).findById(1L);
    }
}

2.3.2 @MockBean

  • @MockBeanSpring Context에 등록된 빈(bean)을 Mock 객체로 바꿔줍니다.
  • 덕분에 Controller, Service, Repository가 연동된 통합 테스트에서도 유용합니다.
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.boot.test.mock.mockito.MockBean;

import java.util.Optional;

import static org.mockito.Mockito.*;
import static org.junit.jupiter.api.Assertions.*;

@SpringBootTest
class OrderServiceTest {

    @Autowired
    private OrderService orderService;

    @MockBean
    private OrderRepository orderRepository; // Repository 빈을 Mock으로 대체

    @Test
    void should_ReturnOrder_When_OrderExists() {
        // Given
        Order mockOrder = new Order(1L, "ProductA", 3, "CREATED");
        when(orderRepository.findById(1L)).thenReturn(Optional.of(mockOrder));

        // When
        Order result = orderService.getOrderById(1L);

        // Then
        assertNotNull(result);
        assertEquals("ProductA", result.getProductName());
        verify(orderRepository, times(1)).findById(1L);
    }
}

2.3.3 @Spy

  • @Spy는 실제 객체를 사용하면서 특정 메서드만 Mocking할 수 있습니다.
  • 로직의 일부만 제어하고 싶을 때 유용합니다.
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Test;
import org.mockito.Spy;

import static org.mockito.Mockito.*;
import static org.junit.jupiter.api.Assertions.*;

class OrderServiceTest {

    @Spy
    private OrderService orderService = new OrderService();

    @Test
    void should_MockSpecificMethod() {
        // Given
        doReturn(500).when(orderService).calculateTotalPrice(100, 5);

        // When
        int totalPrice = orderService.calculateTotalPrice(100, 5);
        String status = orderService.getOrderStatus(1L); // 실제 메서드 실행

        // Then
        assertEquals(500, totalPrice); // Mocking된 결과 검증
        assertEquals("COMPLETED", status); // 실제 동작 검증

        // Verify
        verify(orderService, times(1)).calculateTotalPrice(100, 5);
        verify(orderService, times(1)).getOrderStatus(1L);
    }
}

2.3.4 @SpyBean

  • @SpyBeanSpring Context에 등록된 실제 빈Spy 객체로 감싸줍니다.
  • 즉, Spring 빈의 실제 동작을 수행하면서 일부 메서드만 Mocking 할 수 있습니다.
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.boot.test.mock.mockito.SpyBean;

import static org.mockito.Mockito.*;
import static org.junit.jupiter.api.Assertions.*;

@SpringBootTest
class OrderServiceIntegrationTest {

    @Autowired
    private OrderService orderService;

    @SpyBean
    private OrderService spyOrderService;

    @Test
    void should_UseSpyBeanToMockSpecificMethod() {
        // Given
        doReturn(1000).when(spyOrderService).calculateTotalPrice(200, 5);

        // When
        int mockedResult = spyOrderService.calculateTotalPrice(200, 5); // Mock 동작
        String status = spyOrderService.getOrderStatus(1L); // 실제 동작 실행

        // Then
        assertEquals(1000, mockedResult); // Mocking 결과
        assertEquals("COMPLETED", status); // 실제 메서드 결과

        // Verify
        verify(spyOrderService, times(1)).calculateTotalPrice(200, 5);
        verify(spyOrderService, times(1)).getOrderStatus(1L);
    }
}
어노테이션 대상 설명 사용 예
@Mock 독립된 가짜 객체 Spring Context와 무관, 가짜 객체 생성 순수 단위 테스트
@MockBean Spring 빈을 가짜로 대체 Spring Context에서 Mock 객체로 교체 Controller와 Service 연결 테스트
@Spy 실제 객체 (부분만 Mocking) 일부 메서드만 Mocking, 나머지는 실제 실행 실제 객체의 부분 테스트
@SpyBean Spring 빈을 Spy 객체로 감싸기 Spring 빈을 감싸서 일부만 Mocking 실제 Spring 빈의 동작과 호출 검증

✨ 2.4 테스트코드를 구조적으로 깔끔하게 짜는 꿀팁

2.4.1 @Nested

import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Nested;
import org.junit.jupiter.api.Test;

import static org.junit.jupiter.api.Assertions.*;

@DisplayName("OrderService 단위 테스트")
class OrderServiceTest {

    private OrderService orderService = new OrderService();

    @Nested
    @DisplayName("주문 생성 테스트")
    class CreateOrderTests {

        @Test
        @DisplayName("정상적으로 주문을 생성한다.")
        void should_CreateOrder_Successfully() {
            Order order = orderService.createOrder("ProductA", 3);
            assertNotNull(order);
            assertEquals("ProductA", order.getProductName());
        }
    }

    @Nested
    @DisplayName("주문 조회 테스트")
    class GetOrderTests {

        @Test
        @DisplayName("존재하는 주문을 조회한다.")
        void should_ReturnOrder_When_OrderExists() {
            Order order = orderService.createOrder("ProductA", 3);
            Order retrievedOrder = orderService.getOrderById(order.getId());
            assertEquals(order, retrievedOrder);
        }
    }
}

2.4.2 @DisplayName

  • @DisplayName은 테스트의 목적이나 동작을 설명하는 이름을 추가
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Test;

import static org.junit.jupiter.api.Assertions.*;

@DisplayName("OrderService 테스트")
class OrderServiceTest {

    private OrderService orderService = new OrderService();

    @Test
    @DisplayName("주문 생성 시 상품 이름과 수량이 올바르게 설정된다.")
    void should_CreateOrder_With_CorrectDetails() {
        Order order = orderService.createOrder("ProductA", 3);
        assertNotNull(order);
        assertEquals("ProductA", order.getProductName());
        assertEquals(3, order.getQuantity());
    }
}

2.4.3 @BeforeEach / @AfterEach

  • @BeforeEach: 각 테스트 실행 전 공통 작업을 수행
  • @AfterEach: 각 테스트 실행 후 정리 작업을 수행
import org.junit.jupiter.api.*;

import static org.junit.jupiter.api.Assertions.*;

class OrderServiceTest {

    private OrderService orderService;

    @BeforeEach
    void setUp() {
        orderService = new OrderService();
    }

    @Test
    @DisplayName("주문 생성 시 상품 이름이 설정된다.")
    void should_CreateOrder_With_ProductName() {
        Order order = orderService.createOrder("ProductA", 3);
        assertEquals("ProductA", order.getProductName());
    }

    @AfterEach
    void tearDown() {
        System.out.println("테스트 정리 작업 실행");
    }
}

2.4.4 @TestMethodOrder + @Order

import org.junit.jupiter.api.*;
import org.junit.jupiter.api.MethodOrderer.OrderAnnotation;

import static org.junit.jupiter.api.Assertions.*;

@TestMethodOrder(OrderAnnotation.class) // @Order 어노테이션 기준으로 순서 지정
@DisplayName("OrderService 순서 지정 테스트")
class OrderServiceTest {

    private static OrderService orderService;

    @BeforeAll
    static void setUp() {
        orderService = new OrderService();
    }

    @Test
    @Order(1)
    @DisplayName("1. 주문을 생성한다.")
    void testCreateOrder() {
        Order order = orderService.createOrder("ProductA", 3);
        assertNotNull(order);
        assertEquals("ProductA", order.getProductName());
    }

    @Test
    @Order(2)
    @DisplayName("2. 주문을 조회한다.")
    void testGetOrder() {
        Order order = orderService.getOrderById(1L); // 1번 ID의 주문 조회
        assertNotNull(order);
        assertEquals("ProductA", order.getProductName());
    }

    @Test
    @Order(3)
    @DisplayName("3. 주문 상태를 변경한다.")
    void testUpdateOrderStatus() {
        Order updatedOrder = orderService.updateOrderStatus(1L, "COMPLETED");
        assertEquals("COMPLETED", updatedOrder.getStatus());
    }
}

2.4.5 @TestMethodOrder(MethodName.class)

import org.junit.jupiter.api.*;
import org.junit.jupiter.api.MethodOrderer.MethodName;

import static org.junit.jupiter.api.Assertions.*;

@TestMethodOrder(MethodName.class) // 메서드 이름 순서대로 실행
class OrderServiceTest {

    @Test
    void a_testCreateOrder() {
        System.out.println("1. 주문 생성");
    }

    @Test
    void b_testGetOrder() {
        System.out.println("2. 주문 조회");
    }

    @Test
    void c_testUpdateOrderStatus() {
        System.out.println("3. 주문 상태 변경");
    }
}
반응형

'Study > etc...' 카테고리의 다른 글

소프트웨어 아키텍처  (0) 2025.01.03
클린 코드  (1) 2024.12.13

+ Recent posts