문제 상황 발생
필드가 많은 엔티티가 있음.
엔티티를 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에 구현하고 엔티티는 깨끗하게 유지, 엔티티의 의존성을 줄이기 위함!