본문 바로가기
JPA

JPA - 값 타입과 불변 객체

by 왈레 2022. 4. 12.

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

댓글