티스토리 뷰
1. 다대일(ManyToOne)
이전 글에서 정리한 내용과 동일
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 프로그래밍 - 기본편(김영한)
'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 |
- Total
- Today
- Yesterday
- clean code
- H2
- 도메인 모델링
- 마이크로서비스 패턴
- 이벤트 스토밍
- Spring Boot
- http
- Git
- MySQL
- 트랜잭셔널 아웃박스 패턴
- 스프링 카프카 컨슈머
- Stream
- 클린코드
- 육각형 아키텍처
- 학습 테스트
- TDD
- ATDD
- mockito
- JPA
- 계층형 아키텍처
- 스프링 예외 추상화
- java8
- kafka
- Ubiquitous Language
- Spring
- 폴링 발행기 패턴
- Spring Data JPA
- HTTP 헤더
- named query
- spring rest docs
일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
1 | 2 | 3 | 4 | 5 | 6 | 7 |
8 | 9 | 10 | 11 | 12 | 13 | 14 |
15 | 16 | 17 | 18 | 19 | 20 | 21 |
22 | 23 | 24 | 25 | 26 | 27 | 28 |
29 | 30 | 31 |