Programming/JPA

[Spring Data JPA] JPA 연관관계 매핑(JOIN)

빈쿵바라기 2022. 8. 2. 19:16

연관관계 매핑이 필요한 이유

팀(Team)과 회원(Member)가 일대다의 관계를 가지는 연관관계일 때 예시 코드

 

팀과 회원의 객체 클래스 코드

@Entity
public class Team {

  @Id @GeneratedValue
  private Long id;
  private String name;

}
  
@Entity
public class Member {

  @Id @GeneratedValue
  private Long id;

  private String name;

  @Column(name = "team_id")
  private Long teamId;
  
  }

팀과 회원을 조회하는 코드

// 1번 ID를 가지는 회원 찾기
Member member = em.findById(1);

// 회원의 TeamId로 팀 찾기
Long teamId = member.getTeamId();
Team team = em.findById(teamId);

위처럼 연관관계가 없다면, 회원과 팀을 찾을 때마다 회원의 TeamId로 계속 팀을 찾아와야하는 번거로움비용이 발생합니다.


테이블의 연관관계와 객체 연관관계(객체 참조)의 차이점

  • 테이블은 외래 키로 조인을 사용해서 연관된 테이블을 찾습니다.
    • 따라서 데이터베이스는 단방향이니 양방향이니 나눌 필요가 없습니다.
  • 객체 참조를 사용해서 연관된 객체를 찾습니다.
    • 객체는 참조용 필드가 있는 객체만 다른 객체를 참조하는 것이 가능합니다.
    • 두 객체 사이에 하나의 객체만 참조용 필드를 갖고 참조하면 단방향 관계
    • 두 객체 모두가 각각 참조용 필드를 갖고 참조하면 양방향 관계
  • 테이블과 객체 사이에는 이런 큰 간격이 있습니다.

연관 관계 정의 규칙

관 관계를 매핑할때는 크게 3가지를 생각해보아야합니다.

  • 방향 : 단방향, 양방향 (객체 참조)
  • 연관 관계의 주인 : 양방향일 때, 연관 관계에서 관리하는 주체
  • 다중성 : 다대일, 일대다, 일대일, 다대다

단방향과 양방향 선택

비즈니스 로직에서 두 객체가 참조가 필요한지 여부로 선택
  • 위에 팀(Team)과 회원(Member)의 예시에서
    • team.getMembers()처럼 참조가 필요하다면 Team → Member 단방향 참조
    • member.getTeam()처럼 참조가 필요하다면 Member → Team 단방향 참조
  • 양방향은 객체 참조에서는 복잡도가 증가하므로 되도록 피하며, 양방향 객체 참조가 필요할 때 만 사용할 것

연관 관계의 주인

양방향 연관 관계일 때 외래키가 있는 객체에 연관관계의 주인으로 지정
  • 연관 관계의 주인이 아닌 객체에서 mappeBy 속성을 사용하여 주인을 지정

다중성

두 객체간의 관계를 나타낼 때 데이터베이스를 기준으로 다중성을 결정
  • @ManyToOne : 다대일(N:1)
  • @ManyToMany : 다대다(N:N)
  • @OneToMany : 일대다(1:N)
  • @OneToOne : 일대일(1:1)

다대일(N:1) 예제 코드

일반적으로 많이 사용하는 다대일(N:1) 연관관계를 연습해보겠습니다. 위에 회원(Member)와 팀(Team) 관계로 예를 들겠습니다.

시나리오
1. 하나의 팀에는 여러 회원이 포함될 수 있습니다.
2. 한명의 회원은 하나의 팀에만 소속될 수 있습니다.
  • 데이터베이스를 기준으로 회원(N) : 팀(1) 연관관계를 가집니다. 
  • 외래키는 회원 테이블이 가지고 있습니다.(대부분 다(N)쪽이 외래키를 갖습니다.)

다대일(N:1) 단방향

@Entity
public class Team {

  @Id @GeneratedValue
  private Long id;
  private String name;

}
  
@Entity
public class Member {

  @Id @GeneratedValue
  private Long id;

  private String name;

  // 다대일 조인
  @ManyToOne
  @JoinColumn(name="team_id")
  private Team team;
  
  }

회원(Member)@ManyToOne로 다대일 연관관계를 알리고, @JoinColumn으로 외래키 컬럼명을 지정하였습니다.

 

다대일(N:1) 양방향

@Entity
public class Team {

  @Id @GeneratedValue
  private Long id;
  private String name;
  
  // 양방향
  @OneToMany(mappedBy="team")
  List<Member> members;

}
  
@Entity
public class Member {

  @Id @GeneratedValue
  private Long id;

  private String name;

  // 다대일 조인
  @ManyToOne
  @JoinColumn(name="team_id")
  private Team team;
  
  }

양방향으로 객체참조를 하려면 다대일 관계속 일(1)인 팀(Team)에 @OneToMany를 추가하고 연관관계의 주인인 mappedBy를 지정해줬습니다. mappedBy에 들어가는 값은 관계 대상이 되는 변수명을 작성하시면 됩니다.