inflearn logo
강의

강의

N
챌린지

챌린지

멘토링

멘토링

N
클립

클립

로드맵

로드맵

지식공유

실전! 스프링 부트와 JPA 활용1 - 웹 애플리케이션 개발

회원 목록 조회

테스트용 데이터, 정적 팩토리 메소드 질문!

613

우와아앙

작성한 질문수 6

0

MVC2편 강의를 참고해서 테스트 데이터를 다음과 같이 만들어보았습니다.

 

 

@Component
@RequiredArgsConstructor
public class TestDataInit {

private final MemberService memberService;


/**
* 테스트용 데이터 추가
*/
@PostConstruct
public void init(){
memberService.join(new Member("test1", new Address("CITY1", "ZIPCODE1", "STREET1")));
memberService.join(new Member("test2", new Address("CITY2", "ZIPCODE2", "STREET2")));
}


}

 

 

Member엔티티

@Entity
@Getter
@Setter
public class Member {

@Id
@GeneratedValue
@Column(name = "member_id")
private Long id;

private String name;

@Embedded
private Address address;

@OneToMany(mappedBy = "member")
private List<Order> orders = new ArrayList<>();

public Member(String name, Address address) {
this.name = name;
this.address = address;
}

public Member() {
}
}

 

위의 코드와 같이 MemberService.join()메소드를 사용하기 위해서 Member 엔티티에서 생성자를 추가하였는데 강의 에서는 엔티티를 최대한 순수하게 유지하는 것이 좋다고 하셨는데 이런 생성자들도 만들지 않는게 좋은 설계일까요?

 setter를 쓰지 않고, order엔티티처럼 생성자에 접근제한을 두어  정적 팩토리 메소드를 통해서  생성하는게 좋다고 하셔서 Member는 그대로 두고, Book엔티티를 아래와 같이 바꾸어 보았습니다.

Item과 Book처럼 추상클래스의 상속관계는 setter를 사용하지 않고, 어떤식으로 코드를 짜야할지 감이 안와서 아래와 같이 무작정 setter를 빼고 만들어 봤습니다.

 

@Component
@RequiredArgsConstructor
public class TestDataInit {

private final MemberService memberService;
private final ItemService itemService;





/**
* 테스트용 데이터 추가
*/
@PostConstruct
public void init(){
memberService.join(new Member("test1", new Address("CITY1", "ZIPCODE1", "STREET1")));
memberService.join(new Member("test2", new Address("CITY2", "ZIPCODE2", "STREET2")));
itemService.saveItem(Book.createItem("JPA1", 10000, 100, "test1", "test1"));
itemService.saveItem(Book.createItem("JPA2", 20000, 200, "test2", "test2"));

}
}

 

 

 

Item엔티티

@Entity
@Inheritance(strategy = InheritanceType.SINGLE_TABLE)
@DiscriminatorColumn(name = "dtype")
@Getter
@NoArgsConstructor(access = AccessLevel.PROTECTED)
public abstract class Item {

@Id
@GeneratedValue
@Column(name = "item_id")
private Long id;

private String name;
private int price;
private int stockQuantity;

@ManyToMany(mappedBy = "items")
private List<Category> categories = new ArrayList<>();

protected Item(String name, int price, int stockQuantity) {
this.name = name;
this.price = price;
this.stockQuantity = stockQuantity;
}

//비즈니스 로직

/**
* stock 증가
*/
public void addStock(int quantity){
this.stockQuantity += quantity;
}

/**
* stock 감소
*/
public void removeStock(int quantity){
int restStock = this.stockQuantity - quantity;
if (restStock < 0) {
throw new NotEnoughStockException("need more stock");
}
this.stockQuantity = restStock;
}


}

 

Book엔티티에서 super로 참조하기 위해서 Item에 생성자를 추가하고, 접근제한을 PROTECTED로 변경했습니다.

 

 

 

Book엔티티

@Entity
@DiscriminatorValue("B")
@Getter
@NoArgsConstructor(access = AccessLevel.PRIVATE)
public class Book extends Item {

private String author;

private String isbn;


private Book(String name, int price, int stockQuantity, String author, String isbn) {
super(name, price, stockQuantity);
this.author = author;
this.isbn = isbn;
}

public static Book createItem(String name, int price, int stockQuantity, String author, String isbn) {
Book book = new Book(name,price,stockQuantity,author,isbn);

return book;
}
}

 

위의 코드와 같이 생성자들의 접근제한을 private으로 두고 createItem의 정적 팩토리 메소드를 생성했습니다.

 

 

ItemController

 

@PostMapping("/items/new")
public String create(BookForm form) {
Book book = Book.createItem(form.getName(), form.getPrice(), form.getStockQuantity(), form.getAuthor(), form.getIsbn());
itemService.saveItem(book);
return "redirect:/items";

}

 

추상클래스의 상속관계에서 정적 팩토리 메소드를 작성하는 방법이 맞는가 싶네요..

이런식으로 사용하는게 맞는 방법인가요?? 뭔가 짜다보니 이상한것 같기도해서요  너무 두서없이 작성해서 죄송합니다 ㅠㅠ

 

JPA spring-boot java 웹앱 spring

답변 1

2

David

안녕하세요. 우와아앙님, 공식 서포터즈 David입니다.

.
순수하게 유지한다는 말은 꼭 필요한 것들만 있어야 한다는 말입니다. 객체를 생성하기 위해 생성자가 필요하다면 사용하시면 됩니다. setter를 사용하지 않고 생성시 필요한 값들을 넘겨줘서 생성할 수 있는 생성자면 더욱 좋겠죠.

상속관계에서도 마찬가지입니다. Item에도 생성자가 있고, Book에도 생성자가 있을텐데

Book 생성자 내부에서 Item 생성자를 호출해주면 됩니다.

public Book(int price, String authorName) {

  super(price);

  this.authorName = authorName;

}

그리고 필요하다면 생성자를 정적 팩토리 메서드로 변경하는 것을 고려해볼 수도 있습니다.

자세한 내용은 아래 글을 참고부탁드립니다.

https://velog.io/@ljinsk3/%EC%A0%95%EC%A0%81-%ED%8C%A9%ED%86%A0%EB%A6%AC-%EB%A9%94%EC%84%9C%EB%93%9C%EB%8A%94-%EC%99%9C-%EC%82%AC%EC%9A%A9%ED%95%A0%EA%B9%8C
.
감사합니다.

OrderServiceTest 상문주문 테스트 시 update 쿼리 문의

0

28

1

sdk 설정 오류

0

69

2

오탈자 - @Transactional

0

63

1

src/test/resources 테스트 경로 문제

0

63

1

상품 등록후 H2 db 출력 순서 바꿀 수 있나요?

0

70

1

MemberRepositoryTest 실행오류

0

90

1

boot 4.x >>> trasasction rolled back log & p6spy(영한님, 수업 자료 업데이트 해주시면 감사하겠습니다!!)

1

197

2

강의 마지막 QueryDSL 사용 부분 질문있습니다

1

154

2

클라이언트에서 isbn과 author 수정 요청을 한 경우에 대해 질문드립니다.

0

57

1

도메인 모델 패턴 vs 트랜잭션 스크립트 패턴

0

82

1

기본 생성자

0

67

1

h2 DB 연결시 jdbc url 변경 이유가 궁금합니다.

0

107

1

멤버서비스테스트 부분에서 막힙니다.

0

174

4

실무에서도 EntityManager를 이용해서 많이 작업하는 편일까요?

0

124

1

초반에 h2 다운로드 과정 꼭 필요한가요?

0

129

2

자신 필드에도 get으로 접근하는 이유가 있을까요?

0

122

1

24분 27초 연관관계 편의 메서드 위치

0

118

1

단건 주문만 가능하게 한건 의도한 부분이신가요?

0

114

2

빌드 툴, Gradle

0

65

1

h2연결은 된 것 같은데 엔티티 테이블까지 작성 후 확인해보아도 테이블이 안보입니다

0

82

2

Repository에서 EntityManager 주입 방식 차이

0

97

1

롬복과 사용자 정의 setter 메서드

0

78

1

주문 목록 조회 fetch join 질문드립니다

0

91

1

dirty checking 질문드립니다.

0

87

1