일대일 [1:1]
주 테이블에 외래 키
- 객체지향 개발자들이 선호하는 방식
단방향
@Entity
public class Member {
@Id @GeneratedValue
@Column(name = "MEMBER_ID")
private Long id;
private Strisng username;
@OneToOne
@JoinColumn(name = "LOCKER_ID")
private Locker locker;
...
}
@Entity
public class Locker {
@Id @GeneratedValue
@Column(name = "LOCKER_ID")
private Long id;
private String name;
...
}
양방향
@Entity
public class Member {
@Id @GeneratedValue
@Column(name = "MEMBER_ID")
private Long id;
private Strisng username;
@OneToOne
@JoinColumn(name = "LOCKER_ID")
private Locker locker;
...
}
@Entity
public class Locker {
@Id @GeneratedValue
@Column(name = "LOCKER_ID")
private Long id;
private String name;
@OneToOne(mappedBy = "locker")
private Member member;
...
}
- MEMBER 테이블이 외래 키를 가지고 있으므로 Locker.member에 mappedBy를 선언해서 연관관계의 주인이 아니라고 설정한다.
대상 테이블에 외래 키
단방향
- 일대일 관계 중 대상 테이블에 외래 키가 있는 단방향 관계는 JPA에서 지원하지 않는다.
양방향
@Entity
public class Member {
@Id @GeneratedValue
@Column(name = "MEMBER_ID")
private Long id;
private Strisng username;
@OneToOne(mappedBy = "member")
private Locker locker;
...
}
@Entity
public class Locker {
@Id @GeneratedValue
@Column(name = "LOCKER_ID")
private Long id;
private String name;
@OneToOne
@JoinColumn(name = "MEMBER_ID")
private Member member;
...
}
** 프록시를 사용할 때 외래 키를 직접 관리하지 않는 일대일 관계는 지연 로딩으로 설정해도 즉시 로딩된다.
다대다 [N:N]
- 관계형 데이터베이스는 정규화된 테이블 2개로 다대다 관계를 표현할 수 없다. 그래서 보통 다대다 관계를 일대다, 다대일 관계로 풀어내는 연결 테이블을 사용한다.
- 객체는 테이블과 다르게 객체 2개로 다대다 관계를 만들 수 있다.
다대다: 단방향
@Entity
public class Member {
@Id @Column(name = "MEMBER_ID")
private String id;
private String username;
@ManyToMany
@JoinTable(name = "MEMBER_PRODUCT",
joinColumns = @JoinColumn(name = "MEMBER_ID"),
inverseJoinColumns = @JoinColumn(name = "PRODUCT_ID"))
private List<Product> products = new ArrayList<>();
...
}
@Entity
public class Product {
@Id @Column(name = "PRODUCT_ID")
private String id;
private String name;
...
}
- @JoinTable.name : 연결 테이블을 지정한다.
- @JoinTable.joinColumns : 현재 방향인 회원과 매핑할 조인 컬럼 정보를 지정한다.
- @JoinTable.inverseJoinColumns : 반대 방향인 상품과 매핑할 조인 컬럼 정보를 지정한다.
다대다: 양방향
@Entity
public class Product {
@Id @Column(name = "PRODUCT_ID")
private String id;
private String name;
@ManyToMany(mappedBy = "products")
private List<Member> members;
...
}
- 다대다의 양방향 연관관계는 다음처럼 설정하면 된다.
member.getProducts().add(product);
product.getMembers().add(member);
- 양방향 연관관계는 연관관계 편의 메소드를 추가해서 관리하는 것이 편리하다.
다대다: 매핑의 한계와 극복, 연결 엔티티 사용
- @ManyToMany 매핑을 실무에서 사용하기에는 한계가 있다. => 보통 연결 테이블에 다른 컬럼이 더 필요하다.
- 결국, 연결 테이블을 매핑하는 연결 엔티티를 만들고 이곳에 추가한 컬럼들을 매핑해야 한다. 엔티티 간의 관계도 테이블 관계처럼 다대다에서 일대다, 다대일 관계로 풀어야 한다.
@Entity
public class Member {
@Id @Column(name = "MEMBER_ID")
private Strisng id;
@OneToMany(mappedBy = "member")
private List<MemberProduct> memberProducts;
...
}
@Entity
@IdClass(MemberProductId.class) // 복합 기본키 매핑
public class MemberProduct {
@Id
@ManyToOne
@JoinColumn(name = "MEMBER_ID")
private Member member;
@Id
@ManyToOne
@JoinColumn(name = "PRODUCT_ID")
private Product product;
private int orderAmount;
...
}
public class MemberProductId implements Serializable {
private String member;
private String product;
@Override
public boolean equals(Object o) { ... }
@Override
public int hashCode() { ... }
...
}
** JPA에서 복합 키를 사용하려면 별도의 식별자 클래스를 만들어야 한다. 그리고 엔티티에 @IdClass를 사용해서 식별자 클래스를 지정하면 된다.
복합 키를 위한 식별자 클래스의 특징
- 복합 키는 별도의 식별자 클래스로 만들어야 한다.
- Serializable을 구현해야 한다.
- equals와 hashCode 메소드를 구현해야 한다.
- 기본 생성자가 있어야 한다.
- 식별자 클래스는 public이어야 한다.
- @IdClass를 사용하는 방법 외에 @EmbeddedId를 사용하는 방법도 있다.
* 식별 관계 : 부모 테이블의 기본 키를 받아서 자신의 기본 키 + 외래 키로 사용하는 것
다대다: 새로운 기본 키 사용
- 데이터베이스에서 자동으로 생성해주는 대리 키를 Long 값으로 사용하면 간편하고 거의 영구히 쓸 수 있으며 비즈니스에 의존하지 않는다. 또한 ORM 매핑 시에 복합 키를 만들지 않아도 되므로 간단히 매핑을 완성할 수 있다.
// MemberProduct -> Order
@Entity
public class ORder {
@Id @GeneratedValue
@Column(name = "ORDER_ID")
private Long id;
@ManyToOne
@JoinColumn(name = "MEMBER_ID")
private Member member;
@ManyToOne
@JoinColumn(name = "PRODUCT_ID")
private Product product;
private int orderAmount;
...
}
'공부 기록 > Java' 카테고리의 다른 글
[이펙티브 자바 3/E] 아이템20 - 추상 클래스보다는 인터페이스를 우선하라 (1) | 2023.11.13 |
---|---|
[이펙티브 자바 3/E] 아이템19 - 상속을 고려해 설계하고 문서화하라. 그러지 않았다면 상속을 금지하라 (1) | 2023.11.11 |
[이펙티브 자바 3/E] 아이템18 - 상속보다는 컴포지션을 사용하라 (0) | 2023.09.08 |
[이펙티브 자바 3/E] 아이템17 - 변경 가능성을 최소화하라 (0) | 2023.09.07 |
[JPA] 6장 - 다양한 연관관계 매핑(1) - 다대일, 일대다 (0) | 2023.09.07 |