2024.10.29 업데이트
이 전 포스팅까지는 단방향, 양방향에 대해서 알아보았는데 결론은 테이블 관점은 방향이란 게 없고 자식 테이블에 외래키가 존재하며 그 외래키를 통해 부모 테이블과 조인하여 원하는 데이터를 얻어내거나 반대로 부모 테이블은 본인의 PK를 이용해서 자식 테이블의 외래키와 조인하여 원하는 데이터를 얻을 수 있고 객체 관점은 단방향 또는 양방향 연관관계라는 게 존재하며 양방향 연관관계는 사실 단방향 두 개를 의미하며 설계할 때 양방향이 단 하나도 없어도 애플리케이션에 전혀 문제가 없다는 것. 단방향으로 설계를 끝내고 필요하면 양방향 연관관계를 추가할 것이 결론이었다.
이제는 연관관계를 매핑할 때 ManyToOne, OneToMany, OneToOne, ManyToMany 관계에 대해서 알아보려고 한다.
여기서, ManyToOne(다대일), OneToMany(일대다), OneToOne(일대일), ManyToMany(다대다)는, 앞부분에서 외래키를 관리하는것으로 생각하면 된다. 그래서 다대일이면 '다'쪽에 외래키가 있고, 일대다면 '일'쪽에 외래키가 있게 객체에서 설계한다는 단어이다. DB 설계와는 완전히 상관없는 얘기고 자바와 JPA, 즉, 객체 관점에서 외래키 관리를 앞쪽에서 한다고 생각하면 된다.
⭐️ ManyToOne
ManyToOne은 다대일로 다음과 같은 관계를 말한다.
유저 테이블과 게시물 테이블 두 개가 있을 때, 하나의 유저는 여러 개의 포스팅을 할 수 있다. 그럼 포스트 입장에서는 다대일이 되는 것이다. 그리고 외래키를 관리하는 쪽은 포스트 쪽이 된다. 이 ⭐️테이블의 설계를 그대로 객체 관점에서도 적용할 수 있는 게 다대일이다.⭐️
다음 코드를 보자. 멤버와 주문의 관계는 다대일 연관관계로 적용할 수 있다. 이때 자식 테이블은 주문이 될 것이고 주문 테이블에서 부모 테이블의 기본키를 외래키로 참조한다. 그 설계를 그대로 객체에 적용한다.
@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;
}
@Entity
public class Order {
@Id @GeneratedValue
@Column(name = "ORDER_ID")
private Long id;
@ManyToOne
@JoinColumn(name = "MEMBER_ID")
private Member member;
}
주문 테이블에 멤버 타입의 멤버를 참조한다. 이 @JoinColumn 어노테이션은 멤버와 조인할 때, 어떤것을 가지고 조인할 생각이냐?란 질문에 대한 답변이라고 보면 된다. 당연히 멤버의 ID로 하면 된다. 그리고 하나 더 @ManyToOne 어노테이션이 들어있다. 즉, 주문 입장에서는 다대일 관계가 된다. 그리고 위와 같은 설계를 '다대일 단방향 연관관계'라고 한다. 다대일이니까 '다'쪽에 외래키를 가지고 있고, 한쪽에서만 참조하고 있는 형태.
다대일 양방향 연관관계를 만드려면 ?! 다음과 같은 코드로 만들면 된다.
@Entity
public class Member {
@Id @GeneratedValue
@Column(name = "MEMBER_ID")
private Long id;
@OneToMany(mappedBy = "member")
private List<Order> orders = new ArrayList<>();
private String name;
private String city;
private String street;
private String zipcode;
}
- 참고로, 꽤 중요한 내용인데, 컬렉션은 위 코드(private List<Order> orders = new ArrayList<>();)처럼 필드에서 초기화 하는게 가장 좋다. 우선 여러 가지 이유가 있다.
- null Safety
- 하이버네이트가 엔티티를 영속화할 때, 컬렉션을 한번 자기들이 감싸서 하이버네이트가 제공하는 내장 컬렉션으로 변경을 해버린다. 그래서 컬렉션 내부의 값이 변경이 됐는지 아닌지 이런 체크도 자기들 소스에서 유용하게 처리하고 어떤 정상적인 흐름을 수행할 수 있다. 그런데 만약, 영속시킨 이후에 내가 갑자기 setOrders() 같은 메서드를 호출해서 컬렉션을 초기화하거나 값을 바꾼다면? 그럼 하이버네이트가 컬렉션에 대해 지원하는 기능을 사용 못하게 된다. 이게 진짜 문제다.
- 그래서, 결론적으로 필드레벨에서 컬렉션을 초기화하는게 가장 깔끔하고 안전하다.
@Entity
public class Order {
@Id @GeneratedValue
@Column(name = "ORDER_ID")
private Long id;
@ManyToOne
@JoinColumn(name = "MEMBER_ID")
private Member member;
}
복습하자면, 다대일 양방향 연관관계는 없어도 그만인데 조회의 편리함을 위해서 작성해도 괜찮다고 했다. 테이블 관점에서는 외래키는 주문 테이블만 가지고 있다. 양방향은 연관관계의 주인을 설정해야 하고 주인은 테이블 관점에서 외래키를 가지고 있는 쪽(자식테이블, 위 예제에서는 주문 테이블)이 연관관계의 주인이 되면 된다. 주인이 아닌 쪽은 mappedBy 옵션으로 "나는 연관관계의 주인이 아니고, 주인 객체(Order)의 필드 중 `member` 라는 필드와 연관된 필드입니다." 라고 알려주면 된다.
결론을 미리 여기서 말하면, 다대일 단방향 또는 다대일 양방향으로 설계하는 것이 가장 좋고 일대다, 다대다는 그냥 사용을 하면 안 된다고 생각해라.
OneToMany
일대다는 테이블 관점에서는 존재할 수 없는 연관관계다. 일대다이므로, '일' 쪽에 외래키를 가지고 있는, 즉, 부모 테이블이 외래키를 가지고 있는 경우인데 테이블 관점에서는 존재할 수 없다. 그렇기 때문에 일대다 연관관계는 사용하지 않는 것이 좋다는 결론을 바로 위에 작성했다.
우선 그림을 살펴보면 다음과 같다.
객체 관점에서는 이런 경우가 흔하다고 볼 수 있다. 팀에서만 멤버에 관심이 있고 멤버 입장에서는 팀에 관심이 없을 수 있다. 이게 객체 관점에서는 전혀 문제 될 게 없는데 테이블 관점에서는 있을 수 없다. 1:N 관계에서 언제나 N 쪽에 외래키가 포함되기 때문이다. 아래 사진이 테이블 관점에서의 ERD이다.
그래서 객체 관점에서는 가능한 부분이기 때문에 다음과 같이 작성할 수 있다.
package org.example.entity.mapping;
import javax.persistence.*;
import java.util.ArrayList;
import java.util.List;
@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<>();
}
- @JoinColumn 부터 애매해진다. 외래키를 가지고 있는데 조인하는 필드도 본인의 ID라니! 그렇지만, 이렇게 해야 테이블 관점에서는 동작하기 때문에 어쩔 수 없다.
package org.example.entity.mapping;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.Id;
@Entity
public class Member {
@Id @GeneratedValue
@Column(name = "MEMBER_ID")
private Long id;
private String username;
}
이렇게 작성하면 실제로 동작은 하는데 굉장히 부자연스럽다. 왜냐하면 테이블 관점에서는 팀에서 외래키가 존재하려야 할 수 없는데 지금 객체 관점에서는 존재하게 돼버리니 말이다. 각설하고 위처럼 일대다 단방향으로 설계를 했고 실제로 코드를 돌려보자.
package org.example;
import org.example.entity.Member;
import org.example.entity.RoleType;
import org.example.entity.mapping.Member;
import org.example.entity.mapping.Team;
import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import javax.persistence.EntityTransaction;
import javax.persistence.Persistence;
import java.util.List;
public class Main {
public static void main(String[] args) {
EntityManagerFactory emf = Persistence.createEntityManagerFactory("hello");
EntityManager em = emf.createEntityManager();
EntityTransaction tx = em.getTransaction();
tx.begin();
try {
Member member = new Member();
member.setUsername("member1");
em.persist(member);
Team team = new Team();
team.setName("team1");
team.getMembers().add(member);
em.persist(team);
tx.commit();
} catch (Exception e) {
System.out.println(e.getMessage());
tx.rollback();
} finally {
em.close();
emf.close();
}
}
}
이렇게 코드를 작성하고 돌려보면 문제 없이 테이블이 만들어지고 데이터가 들어갈 거다. 문제없이 잘 들어가는데 다음 SQL문을 보자.
이상하다. 왜 이상하냐면 멤버와 팀을 새로 만들고 persist() 호출을 했으니 INSERT문이 실행되는 건 맞는데 그다음이다. 난 분명 다음 코드처럼 팀의 멤버를 가져와 멤버를 추가했는데 SQL문에서 멤버의 UPDATE 쿼리가 실행된다.
team.getMembers().add(member);
왜냐하면 테이블 관점에서는 절대 외래키를 부모가 관리하지 않기 때문에 결국 멤버를 추가해서 팀과 결합시켰으면 멤버가 속한 팀 FK를 업데이트해줘야 하는 것이다. 이게 이제 문제가 발생할 여지가 크다. 어떤 문제냐면 코드를 보기에는 팀을 가져와 작업을 했는데 쿼리문에서 멤버가 업데이트되니까 이해하기가 힘든 것이다. (설령 JPA를 아주 잘 다루더라도 헷갈릴 소지가 있다 코드가 커지고 시간이 지나면 지날수록)
그러니까 결론은 그냥 다대일 단방향, 양방향을 사용하자...
일대다 양방향도 있는데 그냥 넘어가겠다.. 그걸 알 필요가 없다. 다시 한 번 말하지만 다대일 단방향, 양방향을 사용하자.
OneToOne
이 일대일 관계는 주 테이블이나 대상 테이블 중에 외래키를 선택해서 넣을 수 있는 관계이다. 여기서 주 테이블, 대상 테이블은 약속된 단어가 아니라 자주 사용되는 테이블을 주 테이블이라고 칭하고 그보다 덜 사용되는 테이블을 대상 테이블이라고 말한 것이다.
다만, 외래키에 데이터베이스 유니크 제약조건을 추가해야 한다. (꼭 추가해야 하는 건 아니지만 추가를 하는 게 관리하기 훨씬 유리하다)
일대일은 어떤쪽으로 외래키를 넣어도 상관없다. 위 ERD처럼 Member와 Locker의 일대일 관계에서 멤버에서 외래키를 관리하게 설정하면 된다. 객체 관계에서도 마찬가지로 다음 그림처럼 만들면 된다.
나는 주 테이블을 Member 테이블로 선정하고 (주 테이블은 더 자주 사용되는 테이블이라고 생각하면 된다) 대상 테이블을 Locker 테이블이라고 했을 때 주 테이블에 외래키를 가지는 방식으로 설정해 일대일 단방향 관계를 객체로 구성하면 다음과 같다.
package org.example.entity.mapping;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.Id;
@Entity
public class Locker {
@Id @GeneratedValue
@Column(name = "LOCKER_ID")
private Long id;
private String name;
}
package org.example.entity.mapping;
import javax.persistence.*;
@Entity
public class Member {
@Id @GeneratedValue
@Column(name = "MEMBER_ID")
private Long id;
private String username;
@OneToOne
@JoinColumn(name = "LOCKER_ID")
private Locker locker;
public Long getId() {
return id;
}
public void setId(Long id) {
this.id = id;
}
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
}
일대일 양방향 관계는 다음처럼 현시점은 연관관계의 주인(외래키를 가지고 있는 쪽)이 Member다. 그러니까 Locker에 다음과 같이 코드를 추가한다.
package org.example.entity.mapping;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.Id;
@Entity
public class Locker {
@Id @GeneratedValue
@Column(name = "LOCKER_ID")
private Long id;
private String name;
@OneToOne(mappedBy = "locker")
private Member member
}
이러면 일대일 양방향 관계는 끝이 난다. 당연히 양방향이니까 연관관계의 주인을 선택해야 하고 주인이 아닌 Locker 쪽은 mappedBy로 알려주어야 한다.
이제 단방향으로 외래키가 대상 테이블에 존재하는 경우를 살펴보면 다음 그림처럼 생겼다.
이것 또한 가능한데, 한 가지 주의할 점은! 만약 객체 관점에서 주 테이블에 외래키를 가지고 있는데 테이블 관점에서 대상 테이블에 외래키가 존재하는 경우는 JPA는 지원하지 않는다. 즉, 객체 관점에서 Member 객체가 Locker를 참조하고 있는데 테이블 관점에서는 Locker가 MEMBER_ID를 가지고 있는 경우를 말한다. 이런 경우는 JPA에서 지원하지 않으니까 조심해야 한다.
그래서, 혹시라도 테이블 관점에서 대상 테이블에 외래키가 존재할 때, 객체 관점에서 주 객체(테이블)에 대상 객체(테이블)를 참조하고 싶으면 무조건 일대일 양방향으로 설계해야 한다. 이 방법뿐이 없다.
양방향은 위 주 테이블이 Member일 때 했던 것처럼 반대로 해주면 된다.
일대일 단방향, 양방향은 그냥 주 테이블 또는 대상 테이블 둘 중 하나에 외래키가 들어가면 끝나는 건데 여기서 좀 짚고 넘어갈 부분이 있다.
⭐️ 일대일 양방향 연관관계에서 외래키가 대상 테이블 일 때는 지연 로딩처리가 불가능하다.
이를 이해하기 위해 주 테이블에 외래키가 있을 때를 살펴보자.
주 테이블에 외래키를 보관하는 설계를 했을 때(양방향, 단방향 상관없음) 멤버를 조회하면 LOCKER값은 NULL 또는 특정 값이 존재할 거다. 멤버 테이블에 LOKCER_ID가 있고 멤버가 LOCKER가 할당된 상태라면 NULL값이 아니겠고 할당되지 않은 상태라면 NULL일 테니까. 그럼 할당된 상태일 때 지연 로딩이면 그대로 Lazy Fetch를 이행하면 되고 NULL이면 값이 없으니까 그냥 NULL을 반환하면 된다.
근데 대상 테이블에 외래키가 있는 일대일 양방향 연관관계는 그게 불가능하다. 무슨 소리냐면 만약 아래 그림처럼 대상 테이블에 외래키가 있을 때를 살펴보자.
만약 위처럼 대상 테이블에 외래키가 있을 때 멤버를 조회했다고 가정해 보자. 멤버만 조회했을 때 바로 LOCKER와 연결된 상태인지 알 수 있을까? 없다. 멤버 테이블을 봐라. 멤버 테이블엔 LOCKER 관련 외래키가 없다. 그래서 JPA는 결국 LOCKER 테이블을 조회해서 현재 조회한 멤버의 PK와 동일한 멤버 ID가 있는지 확인을 해야 한다. 즉, 어차피 LOCKER를 조회하기 때문에 지연 로딩을 해봤자 아무런 이점을 가질 수 없단 소리다. 그래서 지연 로딩이 의미가 없고 지연 로딩으로 적용을 해도 바로 데이터를 받아온다.
그래서 만약 일대일 관계를 사용하는 경우엔 주 테이블에 외래키를 보관하는 게 데이터베이스 관점 말고 개발자 관점에서 좀 더 유리할 수 있다. 근데 이건 객체 관점이고, 데이터베이스 관점에서는 조금 다르다. 예를 들어, 멤버 테이블에 외래키를 가지도록 테이블 관점에서 설계를 최초에는 했는데, 이후에 갑자기 일대일이 아니라 "멤버가 여러 라커를 가질 수 있다"라는 비즈니스 규칙이 변경되면, 그땐 데이터베이스의 테이블을 수정해야 하는데 그게 만만치가 않기 때문에 (왜냐하면 테이블 관점에서는 1:N에서 N쪽에 외래키가 반드시 있게 되는데 최초 설계는 멤버에 외래키가 있는 상태이니까) DBA는 이렇게 멤버에 외래키가 있는게 싫을 수도 있다. 그래서 이건 서로간의 충분한 협의가 필요하다. 근데 그냥 멤버에 외래키를 제안하고 DBA가 싫다고 하면 그냥 싸우지말고 말 들어주자. 객체입장에서 일대일 양방향으로 만들면 되는데 데 싸우는 시간이 아깝다.
ManyToMany
이건 쓰면 안 된다. 그냥 이게 결론이다. 근데 왜 쓰면 안 되는지를 알아보자.
우선 데이터베이스에서 다대다 관계는 없다. 즉, 다대다 관계를 두 개의 테이블이 하나의 새로운 테이블(그 두 테이블을 연결해 주는)과 1:N 관계로 만들어야 한다. 아래 그림처럼 말이다.
근데 이게 왜 안되냐면, JPA에서 중간 테이블에 필요한 데이터가 들어가질 않는다. 예를 들어 위 상황에서 주문 수량이나 주문 일자 같은 데이터가 필요할 수 있는 가능성이 농후한데 그 데이터를 연결 테이블이 가질 수 없다. 아무것도 넣지 못한다. 각 테이블의 외래키 말고는.
그래서 다대다를 해결하는 방법은 연결 테이블을 새로운 엔티티로 승격시켜 사용하는 것. 다음 그림처럼 말이다.
그래서 위처럼 연결 테이블(MemberProduct)을 엔티티(ORDER)로 승격시켜 다대다를 일대다, 다대일로 풀어라.
그리고, 데이터베이스에서는 이렇게 풀 때, 중간 테이블의 PK를 양 옆 두 테이블의 FK를 하나의 PK로 만드는 경우가 종종 있는데 그러지 말고 그냥 이 테이블만의 고유 PK(ORDER_ID)를 만드는게 운영면에서 훨씬 유리하다.
그럼 예시를 들어보자. 자주 사용되는 Follower, Following 같은 경우는 어떻게 해야 할까? 느낌은 ManyToMany인데 그렇게 사용하지 않기로 했다. 이 역시 마찬가지로 엔티티 하나를 승격시켜서 사용하면 된다. 다만, 여기서는 OneToMany, ManyToOne으로 풀 게 아니다. 왜냐하면 테이블이 두 개가 아니라 같은 멤버 테이블 하나일 테니까. 그래서 OneToMany, OneToMany로 풀면 된다.
이런 형태로 만들 수 있겠다. 코드를 작성해 보자.
package org.example.entity.mapping;
import javax.persistence.*;
import java.util.ArrayList;
import java.util.List;
@Entity
public class Member {
@Id @GeneratedValue
@Column(name = "MEMBER_ID")
private Long id;
private String username;
@OneToMany(mappedBy = "from")
private List<Followers> following = new ArrayList<>();
@OneToMany(mappedBy = "to")
private List<Followers> followers = new ArrayList<>();
}
package org.example.entity.mapping;
import javax.persistence.*;
@Entity
public class Followers {
@Id @GeneratedValue
private Long id;
@ManyToOne
@JoinColumn(name = "FOLLOW_ID")
private Member from;
@ManyToOne
@JoinColumn(name = "FOLLOWER_ID")
private Member to;
}
⭐️팔로워라는 다대다 테이블을 엔티티로 승격시킨 후, from, to 유저를 @ManyToOne으로 선언한다. 이제 이 외래키를 관리하는 쪽이 연관관계의 주인이 된다. 그럼 반대쪽 멤버 테이블에서는 @OneToMany로 following(팔로우 하는 사람), followers(나를 팔로우하는 사람)를 읽기 전용(mappedBy)으로 선언하면 된다. mappedBy는 반드시 이 애노테이션이 달린 필드로 선언된 엔티티의 실제 필드 이름을 가져야 하고, @JoinColumn의 name은 의미있는 이름을 직접 만들어주어도 괜찮다(FK를 다른 이름으로 만들어도 상관없으니까).⭐️
실전 예제
그러면, 지금까지 배운 모든 내용을 토대로 한번 엔티티 설계를 해보자. ERD는 다음과 같다.
논리적 설계는 이렇게 생겼다.
엔티티의 상세 내용은 다음과 같다.
이 모양을 토대로, 엔티티를 JPA를 활용해서 만들어보자. 참고로, 저기서 다대다도 있다. 다대다는 당연히 사용하면 안된다! 그런데, JPA 스펙에서 지원은 하는 내용이니까 이번 실전 예제에서만 한번만 다뤄보기로 하자.
Member
package cwchoiit.relationship;
import javax.persistence.*;
import java.util.ArrayList;
import java.util.List;
@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<Orders> orders = new ArrayList<>();
}
Orders
package cwchoiit.relationship;
import javax.persistence.*;
import java.time.LocalDateTime;
@Entity
public class Orders {
@Id
@GeneratedValue
@Column(name = "ORDER_ID")
private Long id;
private LocalDateTime orderDate;
private Status status;
@ManyToOne
@JoinColumn(name = "MEMBER_ID")
private Member member;
@OneToOne
@JoinColumn(name = "DELIVERY_ID")
private Delivery delivery;
}
Delivery
package cwchoiit.relationship;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.Id;
@Entity
public class Delivery {
@Id
@GeneratedValue
@Column(name = "DELIVERY_ID")
private Long id;
private String city;
private String street;
private String zipCode;
private Status status;
}
OrderItem
package cwchoiit.relationship;
import javax.persistence.*;
@Entity
public class OrderItem {
@Id
@GeneratedValue
@Column(name = "ORDERITEM_ID")
private Long id;
private Integer orderPrice;
private Integer count;
@ManyToOne
@JoinColumn(name = "ITEM_ID")
private Item item;
@ManyToOne
@JoinColumn(name = "ORDER_ID")
private Orders orders;
}
Item
package cwchoiit.relationship;
import javax.persistence.*;
import java.util.ArrayList;
import java.util.List;
@Entity
public class Item {
@Id
@GeneratedValue
@Column(name = "ITEM_ID")
private Long id;
private String name;
private Integer price;
private Integer stockQuantity;
@ManyToMany(mappedBy = "items")
private List<Category> categories = new ArrayList<>();
}
Category
package cwchoiit.relationship;
import javax.persistence.*;
import java.util.ArrayList;
import java.util.List;
@Entity
public class Category {
@Id
@GeneratedValue
@Column(name = "CATEGORY_ID")
private Long id;
private String name;
@ManyToOne
@JoinColumn(name = "PARENT_ID")
private Category parent;
@OneToMany(mappedBy = "parent")
private List<Category> child = new ArrayList<>();
@ManyToMany
@JoinTable(name = "CATEGORY_ITEM",
joinColumns = @JoinColumn(name = "CATEGORY_ID"), inverseJoinColumns = @JoinColumn(name = "ITEM_ID"))
private List<Item> items = new ArrayList<>();
}
이렇게 만들면 된다. 재밌는 부분은 Category의 경우, 카테고리는 상위 카테고리 라는게 존재하는게 일반적이다. 그래서 parent와 child가 모두 자기 자신의 타입으로 되어 있고, 각자 카테고리가 부모와 자식 카테고리를 가질테니 parent, child가 있다.
그리고, @ManyToMany는 사용하면 안된다. 그러나 한번만 해보기위해서 이렇게 사용했다. 다대다는 @JoinTable 애노테이션을 사용해서 중간 테이블이 하나 생성된다고 했다. 그리고 그 테이블은 다른 필드 하나 없이 양 옆 테이블의 PK만을 가질 수 있다. 그래서 joinColumns, inverseJoinColumns의 값으로 Category와 Item 두 PK를 선언한다.
이렇게 해보면 되는데 역시 다대다는 사용하면 안된다고 했으니 이를 중간 테이블을 엔티티로 승격시켜서 다대일, 일대다로 풀어보자.
Category
package cwchoiit.relationship;
import javax.persistence.*;
import java.util.ArrayList;
import java.util.List;
@Entity
public class Category {
@Id
@GeneratedValue
@Column(name = "CATEGORY_ID")
private Long id;
private String name;
@ManyToOne
@JoinColumn(name = "PARENT_ID")
private Category parent;
@OneToMany(mappedBy = "parent")
private List<Category> child = new ArrayList<>();
}
CategoryItem
package cwchoiit.relationship;
import javax.persistence.*;
import java.time.LocalDateTime;
@Entity
public class CategoryItem {
@Id
@GeneratedValue
private Long id;
@ManyToOne
@JoinColumn(name = "CATEGORY_ID")
private Category category;
@ManyToOne
@JoinColumn(name = "ITEM_ID")
private Item item;
private LocalDateTime created;
}
Item
package cwchoiit.relationship;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.Id;
@Entity
public class Item {
@Id
@GeneratedValue
@Column(name = "ITEM_ID")
private Long id;
private String name;
private Integer price;
private Integer stockQuantity;
}
이렇듯, Item과 Category 사이에 있는 테이블을 엔티티로 승격시킨 CategoryItem이 존재하게 되고, 이 엔티티가 Item과 Category와 다대일, 일대다 관계를 가지도록 변경하면 된다. 원한다면 양방향으로 만들어도 상관없다.
'JPA(Java Persistence API)' 카테고리의 다른 글
[JPA] Part 9. @MappedSuperclass (0) | 2023.10.22 |
---|---|
[JPA] Part 8. 상속관계 매핑 (2) | 2023.10.22 |
[JPA] Part 6. 객체 지향형 모델링, 단방향 양방향 연관관계 주인 (0) | 2023.10.18 |
[JPA] Part 5. 객체와 테이블 매핑 (4) | 2023.10.17 |
[JPA] Part 4. 영속성 컨텍스트 (0) | 2023.10.17 |