1. 값 타입과 불변 객체
왜 복잡하게 값 타입을 나눴을까?
값 타입은 복잡한 객체 세상을 조금이라도 단순화하려고 만든 개념이다.
따라서 값 타입은 단순하고 안전하게 다룰 수 있어야 한다.
개발자는 Entity에 대해서 신경을 많이 쓰지만 값에 대해서는 크게 신경쓰지 않는다.
그 이유는 값 타입이 안전하게 설계가 되있기 때문이다.
값 타입 공유 참조
- 임베디드 타입 같은 값 타입을 여러 엔티티에서 공유하면 위험하다.
- 부작용(side effect) 발생
임베디드 타입을 여러 엔티티에서 공유할 시 부작용 예)
main{
Address address = new Address("city", "street", "1000");
Member member1 = new Member();
member.setUsername("member1");
member.setHomeAddress(address);
em.persist(member1);
Member member2 = new Member();
member.setUsername("member2");
member.setHomeAddress(address); // member1과 같은 address를 사용하고 있다.
em.persist(member2);
member1.getHomeAddress().setCity("newCity");
//member1의 address city를 변경하지만
//member2의 address city도 같이 변경된다.
}
member1의 address를 바꿨는데 member2의 address도 같이 변경되었다.
값 타입 복사
- 값 타입의 실제 인스턴스인 값을 공유하는 것은 위험
- 대신 값(인스턴스)를 복사해서 사용
main{
Address address = new Address("city", "street", "1000");
Member member1 = new Member();
member.setUsername("member1");
member.setHomeAddress(address);
em.persist(member1);
Address copyAddress = new Address(
address.getCity(),
address.getStreet(),
address.getZipcode());
Member member2 = new Member();
member.setUsername("member2");
member.setHomeAddress(copyAddress); //address를 복사한 객체를 넣어주고 있다.
em.persist(member2);
member1.getHomeAddress().setCity("newCity"); //member1의 address city만 변경된다.
}
객체 타입의 한계
- 항상 값을 복사해서 사용하면 공유 참조로 인해 발생하는 부작용 을 피할 수 있다.
- 문제는 임베디드 타입처럼 직접 정의한 값 타입은 자바의 기본 타입이 아니라 객체 타입이다.
- 기본타입은 대입연산시 값이 복사되어 대입되고, 객체 타입은 객체의 값이 아니라 주소가 대입된다.
- 객체 타입은 참조 값을 직접 대입하는 것을 막을 방법이 없다. 즉 객체의 공유 참조는 피할 수 없다.
//기본타입(primitive type)
int a = 10;
int b = a;
b = 4;
출력결과
a = 10
b = 4
//객체 타입
Address a = new Address("Old");
Address b = new a; //객체 타입은 참조를 전달
b.setCity("New");
출력결과
a = new
b = new
불변 객체
- ★★★객체 타입을 수정할 수 없게 만들면 부작용을 원천 차단★★★
- ★★★값 타입은 불변 객체(immutable object)로 설계해야함★★★
- ★★★불변 객체: 생성 시점 이후 절대 값을 변경할 수 없는 객체★★★
- ★★★생성자로만 값을 설정하고 수정자(Setter)를 만들지 않으면 됨 (private setter를 만드는것도 방법)★★★
- 참고: Integer, String은 자바가 제공하는 대표적인 불변 객체
불변이라는 작은 제약으로 부작용이라는 큰 재앙을 막을 수 있다.
값을 변경하고 싶다면?
통으로 바꿔끼어야한다.
main{
Address address = new Address("city", "street", "1000");
Member member = new Member();
member.setUsername("member");
member.setHomeAddress(address);
em.persist(member);
Address newAddress = new Address("NewCity", address.getStreet(), address.getZipcode());
member1.setHomeAddress(newAddress);
}
※출처 - 인프런 김영한 JPA
'JPA' 카테고리의 다른 글
JPA - 값 타입의 비교 (0) | 2022.05.02 |
---|---|
JPA - 기본값 타입과 임베디드 타입 (0) | 2022.04.12 |
JPA - 영속성 전이(CASCADE)와 고아 객체 (2) | 2022.04.11 |
JPA - 즉시 로딩과 지연 로딩 (0) | 2022.04.11 |
JPA - 프록시 (0) | 2022.04.11 |
댓글