[프로그래밍] 엔티티에서 DTO변환

maxxyoung·2023년 2월 12일

문제 상황 발생

필드가 많은 엔티티가 있음.
엔티티를 DTO객체, DTO를 엔티티로 만들 때 빌더 패턴을 사용해서 변환하니
소스가 너무 많아지고 가독성이 떨어짐.
ex.

Customer customer = Customer.builder()
                .email("cc@ccc.com")
                .phoneNumber("3333-3333")
                .customerStatus(CustomerStatus.IN)
                .addresses(new ArrayList<>())
                .regDate(LocalDateTime.now())
                .build();

그럼 이 코드를 생성자를 쓰는 것처럼 한 줄로 줄여볼 수는 없을까?

-> 정적 팩토리 메서드 사용!
-> MapStruct 사용!

정적 팩토리 메서드의 사용

  • 개념
    • 클래스의 인스턴스를 반환하는 단순한 정적 메서드
  • 장점
    • 이름을 가질 수 있다.
      • 하나의 시그니처로 하나의 생성자만 만드는 것이 가능. 정적 팩토리 메서드를 사용하면 적절한 함수명을 사용해 제약을 벗어날 수 있음.
    • 호출될 때마다 인스턴스를 새로 생성하지는 않아도 된다.
      • 불변 클래스는 인스턴스를 미리 만들어 놓거나 새로 생성한 인스턴스를 캐싱하여 재활용하는 식으로 불필요한 객체 생성을 피할 수 있음.
    • 반환 타입의 하위 타입 객체를 반환할 수 있는 능력이 있다.
    • 입력 매개변수에 따라 매번 다른 클래스의 객체를 반환할 수 있다.
    • 정적 팩토리 메서드를 작성하는 시점에는 반환할 객체의 클래스가 존재하지 않아도 된다.
  • 단점
    • 상속을 하려면 public이나 protected 생성자가 필요하니 정적 패토리 메서드만 제공하면 하위 클래스를 만들 수 없다.
    • 정적 팩토리 메서드는 프로그래머가 찾기 어렵다.
  • 명명 방식
    • from: 매개변수를 하나 받아서 해당 타입의 인스턴스를 반환하는 형변환 메서드
    • of: 여러 매개변수를 받아 적합한 타입의 인스턴스를 반환하는 집계 메서드
    • vlaueOf: from과 of의 더 자세한 버전
    • instance, getInstance: (매개변수를 받는다면) 매개변수로 명시한 인스턴스를 반환하지만, 같은 인스턴스임을 보장하지 않음
    • create, newInstance: instance 혹은 newInstance와 같지만, 매번 새로운 인스턴스를 생성해 반환함을 보장.
    • getType: getInstance와 같으나, 생성할 클래스가 아닌 다른 클래스에 팩토리 메서드를 정의할 때 씀. "Type"은 팩토리 메서드가 반환할 객체의 타입.
    • newType: newInstance와 같으나, 생성할 크래스가 아닌 다른 클래스에 팩터리 메서드를 정의할 때 씀. "Type"은 팩토리 메서드가 반환할 객체의 타입임.
    • type: getType과 newType의 간결한 버전

MapStruct 사용

  • MapStruct를 라이브러리를 사용하여 관련 인터페이스를 만들어주면 엔티티 -> Dto로 변환가능
  • 단점으로는 관련 인터페이스를 만들어주어야 하기 때문에 관리되는 클래스들이 많아짐

내가 해결한 방법은?

  • 관리되는 클래스들을 늘리고 싶지 않았기 때문에 dto에 정적 팩토리 메서드 + 스트림을 이용하여 변환하는 방법을 선택
  • DTO 안에 정적 팩토리 메서드를 구현하고 그 메서드 안에서 빌더패턴을 사용해 객체를 만들어 리턴하고 있음
  • 변환 메서드는 최대한 dto에 구현하고 엔티티는 깨끗하게 유지, 엔티티의 의존성을 줄이기 위함!
profile
오직 나만을 위한 글. 틀린 부분 말씀해 주시면 감사드립니다.

0개의 댓글