JPA

[JPA] 임베디드 타입

나는시화 2024. 1. 16. 12:24

임베디드 타입

  • 새로운 값 타입을 직접 정의할 수 있음
  • JPA는 임베디드 타입(embedded type)이라 함
  • 주로 기본 값 타입을 모아서 맏늘어서 복합 값 타입이라고도 함 
  • int, String과 같은 값 타입

임베디드 타입 사용법

  • @Embeddable: 값 타입을 정의하는 곳에 표시 
  • @Embedded: 값 타입을 사용하는 곳에 표시 
  • 기본 생성자 필수 

임베디드 타입의 장점

  • 재사용
  • 높은 응집도
  • Period.isWord()처럼 해당 값 타입만 사용하는 의미 있는 메소드를 만들 수 있음 
  • 임베디드 타입을 포함한 모든 값 타입은, 값 타입을 소유한 엔티티에 생명주기를 의존함  

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

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

@AttributeOverride: 속성 재정의

  • 한 엔티티에서 같은 값 타입을 상요하면?
  • 컬럼 명이 중복됨
  • @AttributeOverrides, @AttributeOverride를 상요해서 컬럼 명 속성을 재정의 

Member Entity

package jbabook.jpa.shop.domain;

import jakarta.persistence.*;

import java.time.LocalDateTime;

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

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

    //주소
    private String city;
    private String street;
    private String zipcode;

    public Long getId() {
        return id;
    }
    public void setId(Long id) {
        this.id = id;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
}

 

Member Entity에 있는 주소와 기간을 Period와 Address로 바꿔준다. 

수정 후 Member Entity 

package jbabook.jpa.shop.domain;

import jakarta.persistence.*;

import java.time.LocalDateTime;

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

    //기간 Period
    @Embedded
    private Period period;
    
    //주소
    @Embedded
    private Address address;

    public Long getId() {
        return id;
    }
    public void setId(Long id) {
        this.id = id;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
}

Period

package jbabook.jpa.shop.domain;

import jakarta.persistence.Embeddable;

import java.time.LocalDateTime;

@Embeddable
public class Period {
    private LocalDateTime startDate;
    private LocalDateTime endDate;

    public LocalDateTime getStartDate() {
        return startDate;
    }

    public void setStartDate(LocalDateTime startDate) {
        this.startDate = startDate;
    }

    public LocalDateTime getEndDate() {
        return endDate;
    }

    public void setEndDate(LocalDateTime endDate) {
        this.endDate = endDate;
    }
}

Address 

package jbabook.jpa.shop.domain;

import jakarta.persistence.Embeddable;

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

    public String getCity() {
        return city;
    }

    public void setCity(String city) {
        this.city = city;
    }

    public String getStreet() {
        return street;
    }

    public void setStreet(String street) {
        this.street = street;
    }

    public String getZipcode() {
        return zipcode;
    }

    public void setZipcode(String zipcode) {
        this.zipcode = zipcode;
    }
}

 


임베디드 타입을 실제로 사용할 때는 setter를 사용하지 않는 것이 좋음. 생성자를 통해서만 값을 지정하는것이 좋다.