Books/도메인 주도 개발 시작하기

Chapter4. 리포지터리와 모델 구현

YoonJong 2024. 4. 8. 15:27
728x90
 

 

모듈 위치

// 인터페이스는 도메인 하위에 위치
domain/
└── repository/
    └── UserRepository.java

// 구현체는 인프라스트럭처 하위에 위치
infrastructure/
├── repository/
	  └── JpaUserRepository.java
  • JPA 를 사용하면 수정한 결과를 반영하는 메서드를 따로 추가하지 않아도 자동으로 DB 에 반영한다 ( 변경 감지 )

  • 스프링 데이터 JPA 예제
@Entity
public class User {

    @Id
    @GeneratedValue
    private Long id;

    private String name;
    private int age;

}

public interface UserRepository extends JpaRepository<User, Long> {

    List<User> findByName(String name);

}

public class UserService {

    @Autowired
    private UserRepository userRepository;

    public void changeName(User user, String newName) {
        user.setName(newName);
        userRepository.save(user);
    }

}

  • 애그리거트(엔티티) 와 밸류 기본 매핑
@Entity
public class Order {

    @Id
    @GeneratedValue
    private Long id;
    private String orderNumber;
    private List<OrderItem> orderItems; // 밸류 타입

}

@Embeddable // 밸류 타입 정의
public class OrderItem {

    private String itemName;
    private int quantity;
    private int price;

}
  • JPA에서 @Entity와 @Embeddable 사용 규칙
    • @Entity 어노테이션은 엔터티 클래스에만 사용
    • 엔터티 클래스는 기본 생성자 필수
    • 엔터티 클래스는 ID 속성을 포함해야 한다.
    • ID 속성은 @Id 어노테이션으로 표시 필수
    • ID 속성은 JPA에서 지원하는 데이터 타입을 사용

  • AttributeConverter : 밸류 타입과 칼럼 데이터 간의 변환을 처리하기 위한 기능 정의

ex ) DB 에 어떤 값을 저장할 때 전/후처리가 필요한 요구가 있을 경우

@Entity
public class User {

    @Id
    @GeneratedValue
    private Long id;

    private String name;

    @Convert(converter = GenderConverter.class) // 사용
    private Gender gender;

}

public class GenderConverter implements AttributeConverter<Gender, String> {

		// Java 객체 속성을 데이터베이스에 저장할 때 사용
    @Override
    public String convertToDatabaseColumn(Gender gender) {
        if (gender == Gender.WOMAN) {
            return "w";
        } else if (gender == Gender.MAN) {
            return "m";
        } else {
            throw new IllegalArgumentException("Unknown gender: " + gender);
        }
    }

		// 데이터베이스에서 읽은 값을 Java 객체 속성으로 변환할 때 사용
    @Override
    public Gender convertToEntityAttribute(String dbData) {
        if ("w".equals(dbData)) {
            return Gender.WOMAN;
        } else if ("m".equals(dbData)) {
            return Gender.MAN;
        } else {
            throw new IllegalArgumentException("Unknown gender code: " + dbData);
        }
    }

}

public enum Gender {
    WOMAN,
    MAN
}

  • 독자적인 라이플 사이클을 갖지 않고 완전히 하나의 엔티티에 의존적일 경우 cascade 와 orphanRemoval 옵션을 사용한다.
    • orphanRemoval 속성을 true로 설정하면 Product 엔터티를 삭제할 때 관련된 Image 엔터티도 자동으로 삭제된다.
@Entity
public class Product {

    @Id
    @GeneratedValue
    private Long id;

    private String name;

    @OneToMany(mappedBy = "product", cascade = CascadeType.ALL, orphanRemoval = true)
    private List<Image> images;

}

@Entity
public class Image {

    @Id
    @GeneratedValue
    private Long id;

    private String filename;

    @ManyToOne
    private Product product;

}

  • 애그리거트의 로딩 전략
    • JPA 매핑을 설정할 때는 애그리거트에 속한 객체가 모두 모여야 완전한 하나가 된다.
  • 즉시로딩 예제
@Entity
public class Member {

    @Id
    @GeneratedValue
    private Long id;

    private String name;

    @OneToMany(mappedBy = "member", fetch = FetchType.EAGER)
    private List<Order> orders;
}

@Entity
public class Order {

    @Id
    @GeneratedValue
    private Long id;

    private String product;

    @ManyToOne
    private Member member;
}

--

// 조회시 N+1 문제가 발생할 수 있다.
Member member = em.find(Member.class, 1L);

// Member 엔터티를 조회하면 Order 엔터티도 모두 함께 조회
// N + 1 쿼리 발생: 1개의 Member 쿼리 + N개의 Order 쿼리
List<Order> orders = member.getOrders();

for (Order order : orders) {
    System.out.println(order.getProduct());
}
  • 지연로딩 예제
@Entity
public class Member {

    @Id
    @GeneratedValue
    private Long id;

    private String name;

    @ManyToOne(fetch = FetchType.LAZY)
    private Team team;
}

@Entity
public class Team {

    @Id
    @GeneratedValue
    private Long id;

    private String name;
}

--

Member member = em.find(Member.class, 1L);

// Member 엔터티를 조회해도 Team 엔터티는 아직 조회되지 않는다.
System.out.println(member.getName());

// Team 엔터티를 실제로 사용할 때 지연 로딩으로 조회된다.
// 1개의 SELECT 쿼리만 발생
System.out.println(member.getTeam().getName());
728x90