목표
- 객체와 테이블 연관관계의 차이를 이해
- 객체의 참조와 테이블의 외래키 매핑
- 용어
- 방향 : 단방향, 양방향
- 다중성 : 1:1, 1:N, N:1, N:M
- 💋연관관계의 주인 : 객체 양방향 연관관계는 관리 주인이 필요
예제
💥객체를 테이블에 맞추어 모델링(연관관계 x) → 안좋다!!!!

멤버
package hellojpa;
import javax.persistence.*;
@Entity
public class Member {
@Id
@GeneratedValue
@Column(name = "member_id")
private Long id;
@Column(name = "username")
private String username;
@Column(name = "team_id")
private Long teamId;
}
팀
@Entity
public class Team {
@Id
@GeneratedValue
@Column(name = "team_id")
private Long id;
private String name;
}
JpaMain ⇒ 외래키 식별자를 직접 다루고 조회 시에도 객체 지향적이지 않다.
try {
//팀 저장
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.getId();
Team findTeam = em.find(Team.class, findTeamId);
tx.commit();
}
💥문제점 : 연관관계 성립 X
객체 지향 모델링 (객체 연관관계 사용)

멤버
@Entity
public class Member {
@Id
@GeneratedValue
@Column(name = "member_id")
private Long id;
@Column(name = "username")
private String username;
/* @Column(name = "team_id")
private Long teamId;*/
@ManyToOne
@JoinColumn(name = "team_id")
private Team team;
}
JpaMain ⇒ 객체 그래프 탐색
try {
//팀 저장
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();
System.out.println("findTeam = " + findTeam.getName());
tx.commit();
}
양방향 매핑
✨사실상, 테이블의 연관관계에서는 방향이 중요치 않다.(Why? FK 하나로 연관관계 확인 가능하다.)

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<>();
}
JpaMain ⇒ 객체 그래프 탐색
try {
//팀 저장
Team team = new Team();
team.setName("TeamA");
em.persist(team);
//회원 저장
Member member = new Member();
member.setUsername("member1");
member.setTeam(team);
em.persist(member);
//영속성 컨텍스트 내용 DB로 저장하고 비우기
em.flush();
em.clear();
/**
* pk를 통해 찾고 싶은 멤버를 조회하고,
* 그 멤버의 팀에 속한 멤버들을 조회하고자 한다.
*/
Member findMember = em.find(Member.class, member.getId());
List<Member> members = findMember.getTeam().getMembers();
for (Member m : members) {
System.out.println("m = " + m.getUsername());
}
tx.commit();
}
mappedBy ⇒ 객체와 테이블이 관계를 맺는 차이에 대해 이해해보자
객체 연관관계 = 2개
회원 → 팀 1개 (단방향)
팀 → 회원 1개 (단방향)
🔔객체의 양방향 관계
객체의 양방향 관계는 사실 서로 다른 단방향 관계 2개(=단방향 연관관계 2개를 만들어야 한다.)

테이블 연관관계 = 1개
회원 ←→ 팀 1개 (양방향)
🔔테이블의 양방향 관계
외래 키 하나로 양방향 관계 성립 (JOIN)
SELECT *
FROM MEMBER M
JOIN TEAM T ON M.TEAM_ID = T.TEAM_ID
SELECT *
FROM TEAM T
JOIN MEMBER M ON T.TEAM_ID = M.TEAM_ID
딜레마

연관관계의 주인(Owner)
양방향 매핑 규칙
mappedBy 사용 불가mappedBy 속성으로 주인 지정💋누구를 주인으로?? (중요)
테이블 상 외래 키가 있는 곳을 주인으로 정해라 (⇒ Member.team이 연관관계의 주인)

양방향 매핑시 가장 많이 하는 실수
⇒ 연관관계의 주인에 값을 입력하지 않음
try {
//팀 저장
Team team = new Team();
team.setName("TeamA");
em.persist(team);
//회원 저장
Member member = new Member();
member.setUsername("member1");
//역방향 관계 설정
team.getMembers().add(member);
**//member.setTeam(team); ****
em.persist(member);
//영속성 컨텍스트 내용 DB로 저장하고 비우기
em.flush();
em.clear();
tx.commit();
}
DB 결과

Why? 읽기 전용으로 쓰인 가짜 매핑 mappedBy가 가리키는 team.getMembers().add(member) 구문때문이다. 주인인 Member.Team에 연관관계를 설정해야 한다. (⇒ 주석 처리된 member.setTeam(team);이 필요)

💥실습 시, 주의할 점
1) 순수 객체 상태를 고려해서 항상 양쪽에 값을 설정하자
team.getMembers().add(member)를 삭제한다면? 2가지 문제가 발생할 수 있다. (물론, 없이도 member.setTeam(team)에 의해 팀에 속한 멤버들을 출력할 수는 있다.)em.flush()와 em.clear()도 없을 때, 영속성 컨텍스트(1차 캐시)에만 데이터가 담긴다. DB에 값이 없다.member.setTeam(team) + team.getMembers().add(member) (= 양방향 둘 다 매핑해주자)2) 연관관계 편의 메소드를 생성하자 (Member의 .setTeam())
/*public void setTeam(Team team) {
this.team = team;
team.getMembers().add(this);
}*/
public void changeTeam(Team team) {
this.team = team;
team.getMembers().add(this);
}
member.changeTeam(team) 하나의 구문으로 양방향 연관관계 설정3) 양방향 매핑시 무한 루프를 조심하자.
toString(), lombok, JSON 라이브러리(Controller에서 Entitiy를 절대 직접 반환하면 안 된다.)양방향 매핑 정리
mappedBy는 필요할 때 사용하자)연관관계 주인 정하는 기준
테이블 구조

객체 구조

💋단방향 연관관계만 설정해도 되지만 양방향 연관관계를 설정하는 이유는 개발 편의상의 이유 또는 조회를 하기 위해서이다. (필요시, 역방향 조회)
멤버 (getter, setter 생략)
@Entity
public class Member {
@Id
@GeneratedValue
@Column(name = "MEMBER_ID")
private Long id;
private String name;
private String city;
private String street;
private String zipcode;
@OneToMany(mappedBy = "member")
private List<Order> orders = new ArrayList<>();
}
주문
@Entity
@Table(name = "ORDERS")
public class Order {
@Id
@GeneratedValue
@Column(name = "ORDER_ID")
private Long id;
private LocalDateTime orderDate;
@Enumerated(EnumType.STRING)
private OrderStatus status;
@ManyToOne
@JoinColumn(name = "MEMBER_ID")
private Member member;
@OneToMany(mappedBy = "order")
private List<OrderItem> orderItems = new ArrayList<>();
public void addOrderItem(OrderItem orderItem) {
orderItems.add(orderItem);
orderItem.setOrder(this);
}
}
주문 상품
@Entity
public class OrderItem {
@Id
@GeneratedValue
@Column(name = "ORDER_ITEM_ID")
private Long id;
@ManyToOne
@JoinColumn(name = "ORDER_ID")
private Order order;
@ManyToOne
@JoinColumn(name="ITEM_ID")
private Item item;
private int orderPrice;
private int count;
}
상품은 그대로.