티스토리 뷰

ORM/JPA

[JPA] 4. 다양한 연관관계 매핑

mandykr 2021. 11. 29. 13:19

 

 

1. 다대일(ManyToOne)

이전 글에서 정리한 내용과 동일

[JPA] 3. 연관관계 매핑

 

2. 일대다(OneToMany)

1) 일대다 단방향

 @Entity
 public class Member {
   @Id @GeneratedValue
   @Column(name = "MEMBER_ID")
   private Long id;
   
   @Column(name = "USERNAME")
   private String name;
   private int age;
   
   // getter, setter
}

@Entity
public class Team {
  @Id @GeneratedValue
  @Column(name = "TEAM_ID")
  private Long id;
  private String name;
  
  @OneToMany // --①
  @JoinColumn(name = "TEAM_ID") // --②
  private List<Member> members = new ArrayList<>();
  
  // getter, setter
}

// main
Member member = new Member();
member.setName("member1");
em.persist(member);

Team team = new Team();
team.setName("team1");

team.getMembers.add(member); // --③
em.persist(team);

① 일대다 관계에서 일(Team)이 연관관계의 주인 -> 다른 테이블(MEMBER)의 외래키를 관리

② Team 의 OneToMany 관계에서 JoinColumn을 사용하지 않으면 JoinTable(TEAM_MEMBER) 방식을 사용함

③ team.getMembers 에 member를 추가할 때 MEMBER 테이블의 TEAM_ID를 UPDATE하는 쿼리 실행 (Team 엔티티를 수정할 때 MEMBER 테이블을 수정)

실무에서 운영하기 어려워 권장하는 방식은 아님.

다대일 양방향 매핑을 사용하자.

 

참고) setter 보다 생성자, builder 패턴을 사용하자

 

2) 일대다 양방향

 @Entity
 public class Member {
   @Id @GeneratedValue
   @Column(name = "MEMBER_ID")
   private Long id;
   
   @Column(name = "USERNAME")
   private String name;
   private int age;
   
   @ManyToOne
   @JoinColumn(insertable=false, updatable=false) // --①
   private Team team;
   
   // getter, setter
}

@Entity
public class Team {
  @Id @GeneratedValue
  @Column(name = "TEAM_ID")
  private Long id;
  private String name;
  
  @OneToMany
  @JoinColumn(name = "TEAM_ID")
  private List<Member> members = new ArrayList<>();
  
  // getter, setter
}

// main
Member member = new Member();
member.setName("member1");
em.persist(member);

Team team = new Team();
team.setName("team1");

team.getMembers.add(member);
em.persist(team);

① 다대일 관계를 설정하고 읽기 전용으로 설정(반대편 Team이 연관관계 주인)

다대일 양방향 매핑을 사용하자.

 

3. 일대일(OneToOne)

1) 일대일 단방향

 @Entity
 public class Member {
   @Id @GeneratedValue
   @Column(name = "MEMBER_ID")
   private Long id;
   
   @Column(name = "USERNAME")
   private String name;
   private int age;
   
   @OneToOne
   @JoinColumn(name = "LOCKER_ID") // --①
   private Locker locker;
   
   // getter, setter
}

@Entity
public class Locker {
  @Id @GeneratedValue
  @Column(name = "LOCKER_ID")
  private Long id;
  private String name;
  
  // getter, setter
}

① 다대일 단방향 매핑과 유사, 일대일 관계의 외래키에 유니크 제약조건을 설정해야 함.

단방향 관계에서 대상 테이블(LOCKER)에 외래키가 있는 경우는 지원하지 않는다. -> 필요한 경우 양방향으로 설정

 

2) 일대일 양방향

@Entity
public class Locker {
  @Id @GeneratedValue
  @Column(name = "LOCKER_ID")
  private Long id;
  private String name;
  
  @OneToOne(mappedBy = "locker") // --①
  private Member member;
  
  // getter, setter
}

① 양방향 매핑시 연관관계 주인 설정

 

일대일 관계에서 외래키를 어느 테이블에 둘 것인지를 결정할 때 고려해볼 점,

 - 추후 일대다로 변경이 가능한 경우를 생각해 외래키를 설정하자.

 - 비지니스 로직에서 자주 select되는 테이블(MEMBER)가 외래키를 컬럼으로 갖고 있으면 성능 등 여러 이점을 갖을 수 있다.

  • 주 테이블에 외래 키
    • 주 객체가 대상 객체의 참조를 가지는 것 처럼
    • 주 테이블에 외래 키를 두고 대상 테이블을 찾음
    • 객체지향 개발자 선호
    • JPA 매핑 편리
    • 장점: 주 테이블만 조회해도 대상 테이블에 데이터가 있는지 확인 가능
    • 단점: 값이 없으면 외래 키에 null 허용
  • 대상 테이블에 외래 키
    • 대상 테이블에 외래 키가 존재
    • 전통적인 데이터베이스 개발자 선호
    • 장점: 주 테이블과 대상 테이블을 일대일에서 일대다 관계로 변경할 때 테이블 구조 유지
    • 단점: 프록시 기능의 한계로 지연 로딩으로 설정해도 항상 즉시 로딩됨

 

4. 다대다(ManyToMany)

 

연결테이블을 엔티티로 만들지 않고 매핑을 시키면 연결 이외에 개발과정에서 많은 문제가 발생한다.

일대다, 다대일 매핑을 사용하자.(다대일 양방향)

MEMBER_ID 와 PRODUCT_ID를 복합키로 사용하기 보다 의존성을 낮춰 ORDER_ID 를 PK로 사용하자.

 

 

 

 

출처

https://www.inflearn.com/course/ORM-JPA-Basic 자바 ORM 표준 JPA 프로그래밍 - 기본편(김영한)

728x90

'ORM > JPA' 카테고리의 다른 글

[JPA] 6. 프록시와 연관관계 관리  (0) 2021.12.14
[JPA] 5. 고급 매핑  (0) 2021.12.01
[JPA] 3. 연관관계 매핑  (0) 2021.11.09
[JPA] 2. 엔티티 매핑  (0) 2021.11.07
[JPA] 1. 내부동작방식  (0) 2021.11.06