[JPA] 임베디드 타입

·2024년 4월 16일
0

JPA

목록 보기
9/17
post-thumbnail

💡JPA의 데이터 타입 분류

📗1. 엔티티 타입

  • @Entity로 정의하는 객체
  • 데이터가 변해도 식별자로 지속해서 추적 가능
    • 회원 엔티티의 키나 나이를 변경해도 식별자로 인식 가능

📗2. 값 타입

  • int, Integer, String처럼 단순히 값으로 사용하는 자바 기본 타입이나 객체
  • 식별자가 없고 값만 있으므로 변경 시 추적 불가
    • 숫자 100을 200으로 변경하면 완전히 다른 값으로 대체

📗값 타입의 분류

  • 기본 값 타입
    • 자바 기본 타입(int, double)
    • 래퍼 클래스(Integer, Long)
    • String
  • 임베디드 타입
    • 복합 값 타입
  • 컬렉션 값 타입

💡임베디드 타입(EmbeddedType)

  • 복합 값 타입을 의미
  • 새로운 값 타입을 직접 정의할 수 있다.
  • JPA는 임베디드 타입(embeddedType)이라 한다.
  • 주로 기본 값 타입을 모아서 만들어서 복합 값 타입이라고도 하며, int, String과 같은 값 타입을 의미한다.

📗임베디드 타입 사용법

  • @Embeddeble : 값 타입을 정의하는 곳에 표시

  • @Embedded : 값 타입을 사용하는 곳에 표시

  • 기본 생성자 필수

    📗임베디드 타입 예시

    Member 엔티티는 이름, 근무일(근무 시작일, 근무 종료일), 주소(도시, 번지, 우편번호) 필드를 가진다.

📌임베디드 타입 적용 전

Member.java

@Entity
public class Member{
    @Id @GeneratedValue
    @Column(name="MEMBER_ID")
    private Long id;

    @Column(name = "USERNAME")
    private String username;

    //기간
    private LocalDateTime startDate;
    private LocalDateTime endDate;

    //주소
    private String city;
    private String street;
    private String zipcode;
}
  • 기존 코드와 동일하다.

📌임베디드 타입 적용 후

Period.java

//값타입 정의
@Embeddable
public class Period {
    private LocalDateTime startDate;
    private LocalDateTime endDate;

    public Period(){}

    public Period(LocalDateTime startDate, LocalDateTime endDate) {
        this.startDate = startDate;
        this.endDate = endDate;
    }
	...
}

Address.java

@Embeddable
public class Address {
    private String city;
    private String street;
    private String zipcode;

    public Address() {

    }

    public Address(String city, String street, String zipcode) {
        this.city = city;
        this.street = street;
        this.zipcode = zipcode;
    }
	...
}
  • 근무시작일(startDate)과 근무종료일(endDate)를 필드로 가지는 Period 클래스를 생성
  • 도시(city), 번지(street), 우편번호(zipcode)를 필드로 가지는 Address 클래스를 생성
  • @Embeddable을 사용하여 임베디드 타입임을 정의

Member.java

@Entity
public class Member{
    @Id @GeneratedValue
    @Column(name="MEMBER_ID")
    private Long id;

    @Column(name = "USERNAME")
    private String username;

    //기간
    @Embedded
    private Period workPeriod;
    
    //주소
    @Embedded
    private Address homeAddress;
	...
}
  • 기존에 필드로 직접 작성했던 근무일, 주소의 필드들을 지우고 @Embedded를 사용하여 Period, Address 클래스를 이용하여 필드를 만들었다.

Member 값 저장

Member member = new Member();
member.setUsername("hello");
member.setHomeAddress(new Address("city","street","1000"));
member.setWorkPeriod(new Period());

em.persist(member);
  • 임베디드 타입을 사용함으로써 아래와 같은 엔티티 구조로 변경됐다.

📗임베디드 타입의 장점

  • 재사용이 가능하다.
    • Period, Address 는 Member외에 다른 곳에서도 사용 가능
  • 높은 응집도를 가진다
  • Period.isWork()처럼 해당 값 타입만 사용하는 의미 있는 메서드를 만들 수 있다.
  • 임베디드 타입을 포함한 모든 값 타입은 값 타입을 소유한 엔티티에 생명주기를 의존한다.
    • 엔티티가 생성될 때 임베디드 타입에도 값이 들어오고, 엔티티 사용이 끝났을 때 값 타입도 함께 사용이 끝난다.

📗임베디드 타입과 테이블 매핑

  • 임베디드 타입은 엔티티의 값일 뿐이다.
  • 임베디드 타입을 사용하기 전과 후에 매핑하는 테이블에는 변화가 없다.
  • 객체와 테이블을 아주 세밀하게 매핑하는 것이 가능
  • 잘 설계한 ORM 애플리케이션은 매핑한 테이블의 수보다 클래스의 수가 더 많다.

📗하나의 엔티티에 같은 임베디드 타입을 사용하고 싶다면? @AttributeOverride

📌@AttributeOverride 적용 전

@Entity
public class Member{
   @Id @GeneratedValue
   @Column(name="MEMBER_ID")
   private Long id;

   @Column(name = "USERNAME")
   private String username;

   //기간
   @Embedded
   private Period workPeriod;
   
   //주소
   @Embedded
   private Address homeAddress;
	
   @Embedded
   private Address oldAddress;	//에러! repeated column in mapping for entity.
   ...
}
  • 하나에 엔티티에서 @Embedded만을 이용하여 같은 임베디드 타입을 사용하려하면 에러가 발생한다.

📌@AttributeOverride 적용 후

Member.java

@Embedded
private Address homeAddress;

@Embedded
@AttributeOverrides({
         @AttributeOverride(name="city",
                 column=@Column(name = "work_city")),
         @AttributeOverride(name="street",
                 column=@Column(name = "work_street")),
         @AttributeOverride(name="zipcode",
                 column=@Column(name = "work_zipcode"))
})
profile
백엔드 개발자를 꿈꿉니다 / 이전 블로그 : https://po-dadak.tistory.com/category

0개의 댓글

관련 채용 정보