본문 바로가기

프로그래밍/JPA

양방향 연관관계

양방향 연관관계

엔티티의 관계가 서로를 참조하게 되면 양방향 연관관계라고 한다.

객체에서는 Member는 Team을 가지고 있고 Team은 일대다 관계여서 여러 Member와 관계를 맺을 수 있으므로 List 컬렉션을 사용한다.
테이블에서는 외래키 하나로 Join을 통해 Member와 Team을 조회 가능하기 때문에 단방향 연관관계와 동일하다.

객체 관계 매핑

@Entity
public class Member {
  @Id @GeneratedValue
  private Long id;

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

  @ManyToOne
  @JoinColumn(name = "TEAM_ID")
  private Team team;
  ...
}
@Entity
public class Team {
  @Id @Column(name = "TEAM_ID")
  private String id;

  private String name;

  @OneToMany(mappedBy="team")
  private List<Member> members = new ArrayList<>();
}

Team에 List로 컬렉션이 추가되었다.
@OneToMany: 일대다 관계를 의미하며 Team에게 Member는 다대일이다. 그리고 mappedBy속성으로 연관관계의 주인이 team임을 명시한다.

연관관계의 주인

  • 양방향 매핑의 규칙
    • 두 객체의 연관관계중 하나를 연관관계의 주인으로 정한다.
    • 연관관계의 주인만이 외래키를 관리할 수 있다.(등록, 수정, 삭제)
    • 주인이 아닌쪽에서는 조회만 가능하다.
  • 연관관계의 주인은 외래키가 있는 곳으로 정해야 한다.
    • 다대일, 일대다 관계에서는 항상 다 쪽이 외래키를 가진다.

양방향 연관관계 매핑 시 주의할 점

  • 연관관계의 주인에 값을 입력해야 한다.
Team team = new Team();
team.setName("TeamA");
em.persist(team);
​
Member member = new Memeber();
member.setName("member1");
member.setTeam(team);
em.persist(member);

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

이 경우 DB에서 회원을 조회하면 TEAM_ID가 null임을 확인할 수 있다.

  • Lombok과 JSON 생성 라이브러리에서 무한루프를 조심하자.
    • Member의 toString 메서드를 호출 시 Team의 toString()에 있는 members에서 각 member의 toString을 또 호출하기 때문에 무한루프가 발생한다.
    • 컨트롤러에서 엔티티를 JSON으로 바꾸는 순간 무한루프에 빠지기도 하고 엔티티를 직접 내보내면 엔티티가 변경될 경우 API 스펙이 바뀌게 되니 DTO로 변환하자

참고문서

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

'프로그래밍 > JPA' 카테고리의 다른 글

단방향 연관관계  (0) 2020.10.15
영속성 컨텍스트란?  (0) 2020.10.12
JPA란?  (0) 2020.07.28