연관관계 매핑 기초

개발자·2022년 1월 3일
0

JPA

목록 보기
4/10
post-thumbnail

연관관계 설정 X

객체를 테이블에 맞추어 데이터 중심으로 모델링하면, 협력 관계를 만들 수 없다.

  • 테이블은 외래 키로 조인을 사용해서 연관된 테이블을 찾는다.
  • 객체는 참조를 사용해서 연관된 객체를 찾는다.

Entity

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

@Entity
public class Team {
  @Id @GeneratedValue
  @Column(name = "TEAM_ID")
  private Long id;
  private String name;
}

Main

//저장
Team team = new Team();
team.setName("TeamA");
em.persist(team);

Member member = new Member();
member.setUsername("member1");
member.setTeamId(team.getId());
em.persist(member);

// 연관관계 설정을 안해주면 객체지향스럽지 않음.
Member findMember = em.find(Member.class, member.getId());
Long findTeamId = findMember.getTeadId();
Team findTeam = em.find(Team.class, findTeamId); 


단방향 연관관계

객체 지향 모델링

객체의 참조와 테이블의 외래 키를 매핑(Member -> Team)

Entity

@Entity
public class Member {
  @Id @GeneratedValue
  @Column(name = "MEMBER_ID")
  private Long id;
  
  @Column(name = "USERNAME")
  private String name;
  
  @ManyToOne // Member가 Many, Team이 One
  @JoinColumn(name = "TEAM_ID")
  private Team team;
}

@Entity
public class Team {
  @Id @GeneratedValue
  @Column(name = "TEAM_ID")
  private Long id;
  private String name;
}

Main (연관관계 저장/조회)

Team team = new Team();
team.setName("TeamA");
em.persist(team);

Member member = new Member();
member.setUsername("member1");
member.setTeam(team); //단방향 연관관계 설정, 참조 저장
em.persist(member);

Member findMember = em.find(Member.class, member.getId());
Team findTeam = findMember.getTeam(); //참조를 사용해서 연관관계 조회


양방향 연관관계

단방향 연관관계에서는 한쪽 방향(Member->Team)으로밖에 갈 수 없었지만 양방향은 양쪽으로 가능(Member<->Team)

Entity

@Entity
public class Member {
  @Id @GeneratedValue
  @Column(name = "MEMBER_ID")
  private Long id;
  
  @Column(name = "USERNAME")
  private String name;
  
  @ManyToOne // Member가 Many, Team이 One
  @JoinColumn(name = "TEAM_ID")
  private Team team;
}

@Entity
public class Team {
  @Id @GeneratedValue
  @Column(name = "TEAM_ID")
  private Long id;
  private String name;
  
  @OneToMany(mappedBy = "team") // 양방향
  private List<Member> members = new ArrayList<>();
}

Main

Team team = new Team();
team.setName("TeamA");
em.persist(team);

Member member = new Member();
member.setUsername("member1");
member.setTeam(team);
em.persist(member);

Member findMember = em.find(Member.class, member.getId());
List<Member> members = findMember.getTeam().getMembers();

for(Member m : members) {
	System.out.println(m.getUserName());
}

객체와 테이블이 관계를 맺는 차이

객체 연관관계 = 2개 : 참조가 양쪽에 있어야함
1) 회원 -> 팀 연관관계 1개(단방향)
2) 팀 -> 회원 연관관계 1개(단방향)
테이블 연관관계 = 1개 : TEAM_ID(FK)를 기준으로 관계가 맺어짐
1) 회원 <-> 팀의 연관관계 1개(양방향)

객체의 양방향 관계

  • 객체의 양방향 관계는 사실 양방향 관계가 아니라 서로 다른 단뱡향 관계 2개다.
  • 객체를 양방향으로 참조하려면 단방향 연관관계를 2개 만들어야 한다.
class A {
    B b;
}

class B {
    A a;
}

테이블의 양방향 연관관계

  • 테이블은 외래 키 하나로 두 테이블의 연관관계를 관리
  • MEMBER.TEAM_ID 외래 키 하나로 양방향 연관관계 가짐(양쪽으로 조인할 수 있다.)

💡 연관관계의 주인(Owner)

양방향 매핑 규칙

  • 객체의 두 관계중 하나를 연관관계의 주인으로 지정
  • 연관관계의 주인만이 외래 키를 관리(등록, 수정) (Member별로 Team 등록/수정)
  • 주인이 아닌쪽은 읽기만 가능 (Team은 Member들 읽어옴)
  • 주인은 mappedBy 속성 사용 X
  • 주인이 아니면 mappedBy 속성으로 주인 지정

누굴 주인으로 할지

  • 외래 키가 있는 있는 곳을 주인으로 정한다. => ManyToOne의 Many쪽!
  • 여기서는 Member.team이 연관관계의 주인

양방향 연관관계 주의

  • 순수 객체 상태를 고려해서 항상 양쪽에 값을 설정
  • 연관관계 편의 메소드를 생성
  • 양방향 매핑시에 무한 루프를 조심(ex. toString(), lombok, JSON 생성 라이브러리)

양방향 매핑시 가장 많이 하는 실수

  • 역방향만 연관관계 설정하는 경우
    member.setTeam(team);을 추가해줘야 함!
Team team = new Team();
team.setName("TeamA");
em.persist(team);

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

//역방향(주인이 아닌 방향)만 연관관계 설정. Member.Team이 null이 되어버림.
team.getMembers().add(member);
  • 메모리가 아닌 1차 캐시값을 조회하는 경우
    flush, clear 안해주면 Select 쿼리가 실행되지 않음
Team team = new Team();
team.setName("TeamA");
em.persist(team);

Member member = new Member();
member.setName("member1");
em.persist(member);
//team.getMembers().add(member);

//em.flush();
//em.clear();

Team findTeam = em.find(Team.class, team.getId());
List<Member> members = findTeam.getMembers();

for(Member m : members) {
	System.out.println(m);
}

해결 방법

Member 엔티티에서 Team 값 세팅 시 양쪽에 설정

public class Member {
    public void setTeam(Team team) {
    	this.team = team; // 단방향
    	team.getMembers.add(this); // 역방향
    }
}

Team 엔티티에서 해줘도 된다.(둘 중 하나만)

public class Team {
    public void addMember(Member member) {
    	members.add(member);
        member.setTeam(this);
    }
}

양방향 매핑 정리

  • 단방향 매핑만으로도 이미 연관관계 매핑은 완료
  • 양방향 매핑은 반대 방향으로 조회(객체 그래프 탐색) 기능이 추가된 것 뿐
  • JPQL에서 역방향으로 탐색할 일이 많음
  • 단방향 매핑을 잘 하고 양방향은 필요할 때 추가해도 됨(테이블에 영향을 주지 않음)

Ref.

[인프런] 자바 ORM 표준 JPA 프로그래밍 - 기본편 (김영한)

profile
log.info("공부 기록 블로9")

0개의 댓글