
Java Persistence API의 약자로, Java 진영의 ORM 기술 표준 명세를 말합니다.
ORM ?
Object-relational mapping의 약자로, 객체와 관계형 DB를 맵핑해줍니다. 객체는 객체대로, 관계형 DB는 관계형 DB대로 설계를 하고, ORM 프레임워크가 서로 다른 부분을 중간에서 맵핑해줍니다. 대중적인 언어에는 대부분 ORM이 존재합니다.

JPA 는 애플리케이션과 JDBC API 사이에서 동작합니다.
JDBC API는 자바에서 데이터베이스에 접속할 수 있도록 하는 자바 API 입니다.
Member 객체를 저장하려면 이 객체를 Member DAO에게 넘기고, Member DAO가 JPA에게 Member 엔티티를 저장해달라고 하면 JPA가 알아서 객체를 분석해 INSERT SQL을 만들어줍니다. 또한 JDBC API를 사용해 DB INSERT 쿼리까지 다 날려줍니다.

조회도 마찬가지로 JPA가 객체를 분석해 SELECT 쿼리를 만들고, JDBC API를 사용해 RESULT SET 매핑, 상속 등의 패러다임 불일치까지 해결합니다. 엔티티 오브젝트까지 완성해 반환합니다.

JPA를 사용하면..

jpa.persist(album);
// SQL 등 나머지는 JPA가 처리
INSERT INTO ITEM ...
INSERT INTO ALBUM ...
Album album = jpa.find(Album.class, albumId);
// SQL 등 나머지는 JPA가 처리
SELECT I.*, A.*
FROM ITEM I
JOIN ALBUM A ON I.ITEM_ID = A.ITEM_ID
// 연관관계 저장
member.setTeam(team);
jpa.persist(member);
// 객체 그래프 탐색
Member member = jpa.find(Member.class, memberId);
Team team = memebr.getTeam();
JPA가 성능 최적화까지 고려해주기 때문에 엔티티 계층을 신뢰할 수 있습니다.
class MemberService {
...
public void process()
Member member = memberDAO.find(memberId);
memebr.getTeam(); // 자유로운 객체 그래프 탐색
member.getOrder().getDelivery();
}
}
member 객체를 통해 자유로운 객체 그래프 탐색이 가능합니다. JPA가 관리하는 객체를 엔티티 라고 합니다.
String memberId = "100";
Member member1 = jpa.find(Member.class, memebrId);
Member member2 = jpa.find(Member.class, memebrId);
member1 == member2; // 같다.
JPA는 동일한 트랙잭션 안에서 조회한 엔티티는 같음을 보장해줍니다. JPA를 통해 memberId=100을 조회하면 member1과 member2는 같다고 나옵니다.
같은 트랙잭션 안에서는 같은 엔티티를 반환합니다. 상황에 따라 다르지만, 조회 성능이 약간 향상됩니다.
String memberId = "100";
Member member1 = jpa.find(Member.class, memebrId); // SQL
Member member2 = jpa.find(Member.class, memebrId); // 캐시
member1 == member2;
2번 회원을 조회했지만 SQL은 한번만 실행됩니다.
옵션을 허용하면 트랙젝션을 커밋할 때까지 INSERT SQL을 모았다가, JDBC BATCH SQL 기능을 사용해서 한번에 SQL을 전송합니다.
transaction.begin(); // [트랜잭션] 시작
em.persist(memberA);
em.persist(memberB);
em.persist(memberC);
// 여기까지 INSERT SQL을 모아서 보낸다.
// 커밋하는 순간 DB에 INSERT SQL을 모아서 보낸다.
transaction.commit(); // [트랜잭션] 커밋
지연 로딩 : 객체가 실제 사용될 때 로딩
즉시 로딩 : JOIN SQL로 한번에 연관된 객체까지 미리 조회

member를 조회할 때 항상 team이 같이 사용된다면 한번에 같이 JOIN으로 조회해서 가지오는게 좋습니다. 이럴 때는 즉시 로딩을 사용하면 됩니다.