1) 무한 반복, 지루한 코드
2) 패러다임의 불일치
객체 vs RDB
객체 → SQL 변환(개발자) → RDB
객체와 RDB 차이
상속 : 객체 o, RDB x
연관관계
데이터 타입
테이터 식별 방법
결국, 진정한 의미의 계층 분할이 어렵다.
JPA : Java Persistence API
자바 ORM 표준
애플리케이션과 JDBC 사이 동작
저장 & 조회
JPA 역사?
JPA는 표준 명세
JPA 사용 이유
❗SQL 중심 개발 → 객체 중심 개발
❗생산성 - JPA와 CRUD
jpa.persist(member)
Member member = jpa.find(memberId)
member.setName("변경 내용")
>> 변경감지jpa.remove(member)
❗유지보수
❗패러다임 불일치 해결
1) JPA와 상속
jpa.persist(album)
Album album = jpa.find(Album.class, albumId);
2) JPA와 연관관계
//저장
member.setTeam(team);
jpa.persist(member);
3) JPA와 객체 그래프 탐색
Member member = jpa.find(Member.class, memberId);
Team team = member.getTeam();
class MemberService {
...
public void process() {
Member member = memberDAO.find(memberId);
member.getTeam(); //자유로운 객체 그래프 탐색
member.getOrder().getDelivery();
}
}
4) JPA와 비교
String memberId = "100";
Member member1 = jpa.find(Member.class, memberId);
Member member2 = jpa.find(Member.class, memberId);
//member1 == member2; //같다.
//동일한 트랜잭션에서 조회한 엔티티는 같음을 보장
❗성능
1) 1차 캐시와 동일성 보장
같은 트랜잭션 안에서는 같은 엔티티를 반환
DB Isolation Level
이 Read Commit이어도 애플리케이션에서 Repeatable Read 보장
String memberId = "100";
Member m1 = jpa.find(Member.class, memberId); //SQL
Member m2 = jpa.find(Member.class, memberId); //캐시
//println(m1 == m2) //true
//SQL 1번만 실행
DB Isolation Level
2) 트랜잭션을 지원하는 쓰기 지연
INSERT
트랜잭션을 커밋할 때까지 INSERT SQL을 모음
JDBC BATCH SQL 기능을 사용해서 한번에 SQL을 전송
transaction.begin(); // [트랜잭션] 시작
em.persist(memberA);
em.persist(memberB);
em.persist(memberC);
//여기까지 INSERT SQL을 데이터베이스에 보내지 않는다.
//커밋하는 순간 데이터베이스에 INSERT SQL을 모아서 보낸다.
transaction.commit(); // [트랜잭션] 커밋
UPDATE
3) 지연 로딩과 즉시 로딩
지연 로딩 : 객체가 실제 사용될 때 로딩(프록시 객체)
즉시 로딩 : JOIN SQL로 한번에 연관된 객체까지 미리 조회
❗데이터 접근 추상화와 벤더 독립성
❗표준
ORM = 객체 + RDB 균형을 잘 맞추기 위해 둘 다 잘해야 한다.