H2 데이터베이스는 이미 설치했기 때문에 넘어가도록 하겠습니다.
아래 링크에 H2 데이터베이스 설치했던 내용이 담겨있는 내용이 있습니다.
메이븐 소개
- https://maven.apache.org/
- 자바 라이브러리, 빌드 관리
- 라이브러리 자동 다운로드 및 의존성 관리
- 최근에는 그레들(Gradle)이 점점 유명 (저도 그레들로 많이 사용했습니다.)
프로젝트 생성
- 자바 8 이상(8 권장)
- 메이븐 설정
- groupId : jpa-basic
- artifactId : ex1-hello-jpa
- version : 1.0.0
intelliJ를 킨 후 new Project에서 maven을 선택한 뒤 위의 그림대로 작성 후 프로젝트를 생성해주시면 됩니다.
코드추가 (pom.xml)
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>jpa-basic</groupId>
<artifactId>ex1-hello-jpa</artifactId>
<version>1.0.0</version>
<dependencies>
<!-- JPA 하이버네이트 -->
<dependency>
<groupId>org.hibernate</groupId>
<artifactId>hibernate-entitymanager</artifactId>
<version>5.3.10.Final</version>
</dependency>
<!-- H2 데이터베이스 -->
<dependency>
<groupId>com.h2database</groupId>
<artifactId>h2</artifactId>
<version>1.4.199</version>
</dependency>
</dependencies>
</project>
JPA 설정하기 - persistence.xml
jpa를 쓰기위해선 몇가지 설정을 해야합니다.
- JPA 설정 파일
- /META-INF/persistence.xml 위치 (표준 위치가 정해져 있습니다.)
- persistence-unit name으로 이름 지정 (JPA이름, DB마다 만든다고 합니다.)
- javax.persistence로 시작 : JPA 표준 속성
- hibernate로 시작 : 하이버네이트 전용 속성
코드 추가 (METE-INF/persistence.xml)
<?xml version="1.0" encoding="UTF-8"?>
<persistence version="2.2"
xmlns="http://xmlns.jcp.org/xml/ns/persistence" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/persistence http://xmlns.jcp.org/xml/ns/persistence/persistence_2_2.xsd">
<persistence-unit name="hello">
<properties>
<!-- 필수 속성 -->
<property name="javax.persistence.jdbc.driver" value="org.h2.Driver"/>
<property name="javax.persistence.jdbc.user" value="sa"/>
<property name="javax.persistence.jdbc.password" value=""/>
<property name="javax.persistence.jdbc.url" value="jdbc:h2:tcp://localhost/~/test"/>
<property name="hibernate.dialect" value="org.hibernate.dialect.H2Dialect"/>
<!-- 옵션 -->
<property name="hibernate.show_sql" value="true"/>
<property name="hibernate.format_sql" value="true"/>
<property name="hibernate.use_sql_comments" value="true"/>
<!--<property name="hibernate.hbm2ddl.auto" value="create" />-->
</properties>
</persistence-unit>
</persistence>
데이터베이스 방언
- JPA는 특정 데이터베이스에 종속 X
- 각각의 데이터베이스가 제공하는 SQL 문법과 함수는 조금씩 다름
- 가변 문자 : MySQL은 VARCHAR, Oracle은 VARCHAR2
- 문자열을 자르는 함수 : SQL 표준은 SUBSTRING(), Oracle은 SUBSTR()
- 페이징 : MySQL은 LINIT, Oracle은 ROWNUM
- 방언 : SQL 표준을 지키지 않는 특정 데이터베이스만의 고유한 기능
- hibernate.dialect 속성에 지정
- H2 : org.hibernate.dialect.H2Dialect
- Oracle 10g : org.hibernate.dialect..Oracle10gDialect
- MySQL : org.hibernate.dialect.MySQL5InnoDBDialect
- 하이버네이트는 40가지 이상의 데이터베이스 방언 지원
애플리케이션 개발
JPA 구동 방식
- Persistence라는 클래스를 통해 설정 정보를 조회합니다.
- 설정 정보를 조회해서 EntityManagerFactory라는 클래스를 만듭니다.
- EntityManagerFactory 클래스에서 필요할 때마다 EntityManager를 찍어냅니다.
실습 - JPA 동작 확인
- JpaMain 클래스 생성
- JPA 동작 확인
객체와 테이블을 생성하고 매핑하기
- @Entity
- JPA가 관리할 객체
- @Id
- 데이터베이스 PK와 매핑
localhost:8082를 url로 검색한 뒤, JDBC URL에 persistence.xml에 입력했던 아래의 코드의 value값을 넣어야지 정상적으로 연결이 됩니다.
<property name="javax.persistence.jdbc.url" value="jdbc:h2:tcp://localhost/~/test"/>
Entity 클래스를 만들고, 트랜잭션을 보내주면 log로 쿼리문을 볼 수 있습니다. 위와 같은 쿼리문이 로그에 출력된다면 정상적으로 jpa가 작동하는 것 입니다. 쿼리문이 로그로 출력이 되는 이유는 persistence.xml 코드에서 옵션 첫번째 줄에 show_sql의 value값을 true로 주었기 때문입니다. format_sql은 로그를 위와같이 가독성이 좋게 출력을 해줍니다. use_sql_comments는 주석된 부분을 로그로 보여줄 것인지 아닌지를 설정해줍니다. h2 콘솔로 확인해보면 입력했던 데이터가 저장된 것을 확인할 수 있습니다.
트랜잭션 정석코드로 변경했습니다. 트랜잭션을 보냈을 때, 에러가 발생할 시 rollback하고 정상적으로 종료되면 EntityManager와 EntityManagerFactory를 종료합니다. 지금은 jpa 정석 코드를 보여주기 위해 적었지만 실제로는 spring 위의 코드가 다 없어지고 em.persist(member)만 적으면 작업을 완료할 수 있습니다.
JPA 조회
조회는 위와 같이 find 메소드를 통해 찾을 수 있습니다. 파라미터로 엔티티 클래스명, PK를 넘겨주면 DB에 저장한 데이터를 가져올 수 있습니다. 가져온 Entity의 값을 출력해보면 이전에 넣었던 데이터가 출력되는 것을 볼 수 있습니다.
JPA 삭제
삭제는 위처럼 remove에 찾은 Entity 클래스 필드를 넣어주면 됩니다.
JPA 수정 (신기함)
수정이 진짜 신기합니다. 마치 자바 컬렉션을 사용하는 것처럼 사용만하면 수정이됩니다. 단지, setter를 활용해 데이터를 수정만해준 뒤 트랜잭션만 보내면 JPA가 자동으로 DB 데이터를 수정해줍니다. 이것이 가능한 이유는 JPA를 통해 Entity를 가져오면 해당 Entity는 JPA가 관리하기 때문입니다. 그리고 JPA가 변경이 되는지 안되는지 트랜잭션을 commit하는 시점에 체크합니다. 그 후 바뀐 데이터가 있을 시 업데이터 쿼리를 작성하여 DB에 날려줍니다.
주의
- EntityManagerFactory는 하나만 생성해서 애플리케이션 전체에서 공유
- 웹 애플리케이션 서비스를 실행한다고하면 EntityManagerFactory는 DB 당 하나만 생성한다고 보면 됩니다.
- EntityManager는 쓰레드간에 공유 X (사용하고 버려야 한다.)
- 고객의 요청에 따라서 사용한 뒤 close, 사용한 뒤 close 반복한다고 보면 됩니다.
- 이유는 후에 이해하면 정리하도록!
- JPA의 모든 데이터 변경은 트랙잭션 안에서 실행
- 정말 중요합니다!
- RDB 데이터 변경은 트랜잭션안에서 할 수 있도록 설계가 되어있기 때문입니다.
JPQL 소개
- 가장 단순한 조회 방법
- EntityManager.find(Entity 클래스명, PK 값)
- 객체 그래프 탐색 (a.getB().getC())
- 나이가 18살 이상인 회원을 모두 검색하고 싶다면?
- JPQL을 사용해야 합니다.
현업에서의 개발고민은 여러 테이블이 정말 많은데 필요하면 조인도해야하고, 원하는 데이터를 최적화 해서 가져와야하고, 필요하면 통계성 쿼리도 날려야하고 이러한 작업들을 JPA에서는 JPQL로 도와준다고 보면 됩니다. 쿼리를 어떻게 쓰지라는 고민이 드실텐데 JPA에서는 대안이 있다고 합니다.
실습 - JPQL 소개
- JPQL로 전체 회원 탐색
- JPQL로 ID가 2 이상인 회원만 검색
- 쿼리문에 where m.id =
- JPQL로 이름이 같은 회원만 검색
- JPQL에 대해 자세한 내용은 객체지향 쿼리에서 학습
JPQL로 전체 회원 탐색
log를 잘 보면 실제 전송한 쿼리랑 다르게 모든 필드를 나열한 것을 볼 수 있습니다.
JPQL
- JPA를 사용하면 엔티티 객체 중심으로 개발할 수 있다.
- 문제는 검색 쿼리
- 검색을 할 때도 테이블이 아닌 엔티티 객체로 탐색
- 모든 DB 데이터를 객체로 변환해서 검색하는 것은 불가능
- 애플리케이션이 필요한 데이터만 DB에서 불러오려면 결국 검색 조건이 포함된 SQL이 필요
- JPQL은 SQL을 추상화한 JPQL이라는 객체 지향 쿼리 언어 제공
- SQL을 추상화했기 때문에 특정 데이터베이스에 의존 X
- SQL 문법과 유사, SELECT FROM, GROUP BY, WHERE, HAVING, JOIN 지원
- JPQL은 엔티티 객체를 대상으로 쿼리
- SQL은 데이터베이스 테이블을 대상으로 쿼리
- JPQL은 뒤에서 아주 자세히 다룸
출처
'spring > 인프런 강의 정리' 카테고리의 다른 글
[JPA 기본편] 4. 엔티티 매핑 (0) | 2022.03.05 |
---|---|
[JPA 기본편] 3. 영속성 관리 - 내부 동작 방법 (0) | 2022.03.05 |
[JPA 기본편] 1. JPA 소개 (0) | 2022.02.19 |
[HTTP 웹 기본 지식] 8. HTTP 헤더 - 캐시와 조건부 요청 (0) | 2022.01.15 |
[HTTP 웹 기본 지식] 7. HTTP 헤더 - 일반 헤더 (0) | 2022.01.13 |