[JPA] 2. JPA 시작

최진민·2021년 6월 4일
0

JPA

목록 보기
2/11
post-thumbnail

Hello JPA - 프로젝트 생성

  • h2 DB 설치 (http://www.h2database.com/)

    • 명령 프롬포트를 통해 설치한 경로에 들어간다.
    • (mac) chmod 755 h2.sh
      • 🧨Windows는 입력 불필요
    • (Windows) h2.bat 입력 ((mac) ./h2.sh 입력) ⇒ 브라우저로 연동 ⇒ 무언가 뜨지 않을 경우 : IP 주소만 localhost로 변경 (ex) http://xxx.xxx.xxx.xxx:8082/http://localhost:8082/)
  • Maven

    • 자바 라이브러리, 빌드 관리
    • 라이브러리 자동 다운, 의존성 관리
    • (최근 Gradle) - 추후 Gradle로 빌드하는 실무 연습 블로깅을 할 것!
  • 프로젝트

    • groubId : jpa-basic

    • artifactId : ex1-hello-jpa

    • 의존성 추가

      <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>
    • JPA 설정 - persistence.xml

      • JPA 설정 파일

      • /META-INF/persistence.xml 위치

      • persistence-unit name으로 이름 지정

      • javax.persistence로 시작: JPA 표준 속성

      • hibernate로 시작: 하이버네이트 전용 속성

        <?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>
      • 데이터베이스 방언(Dialect)

        • JPA는 특정 DB에 종속 X

        • DB 별 제공하는 SQL 문법과 함수는 조금씩 다르다

          • 가변 문자 : MySQL → VARCHAR, Oracle → VARCHAR2
          • 문자열 자르는 함수 : SQL 표준 → SUBSTRING(), Oracle → SUBSTR()
          • 페이징 : MySQL → LIMIT, Oracle → ROWNUM
        • 방언(Dialect) : SQL 표준을 지키지 않는 특정 데이터베이스만의 고유 기능

        • hibernate.dialect 속성에 지정

          • H2 : org.hibernate.dialect.H2Dialect
          • Oracle 10g : org.hibernate.dialect.Oracle10gDialect
          • MySQL : org.hibernate.dialect.MySQL5InnoDBDialect
          • 하이버네이트는 40가지 이상의 데이터베이스 방언 지원

Hello JPA - 애플리케이션 개발

  • JPA 구동 방식

  • 실습 (항상 H2 Database 켜놓기)

    • 0) h2 database 설정은 직전에 설정한 persistence.xml과 동일하게 설정

    • 1) Pesistence 등록 (패키지 & 클래스 네임 확인)

      package hellojpa;
      
      import javax.persistence.EntityManager;
      import javax.persistence.EntityManagerFactory;
      import javax.persistence.Persistence;
      
      public class JpaMain {
          public static void main(String[] args) {
              EntityManagerFactory emf = Persistence.createEntityManagerFactory("hello");
      
              EntityManager em = emf.createEntityManager();
              //code
      
              em.close();
              emf.close();
          }
      }
    • 2) 객체 & 테이블 생성, 매핑

      • 객체(Java)

        package hellojpa;
        
        import javax.persistence.Entity;
        import javax.persistence.Id;
        
        @Entity
        //@Table(name = "USER") // 테이블 이름 매핑! 없으면 클래스 명과 같은 테이블과 매핑되는 관례를 따른다.
        public class Member {
        
            @Id
            private Long id;
        
            //@Column(name = "username") // 컬럼 이름 매핑! 없으면 필드 명과 같은 컴럼과 매핑되는 관례를 따른다.
            private String name;
        
            public Long getId() {
                return id;
            }
        
            public void setId(Long id) {
                this.id = id;
            }
        
            public String getName() {
                return name;
            }
        
            public void setName(String name) {
                this.name = name;
            }
        }
      • 테이블(H2)

        create table Member (
        		id bigint not null,
        		name varchar(255),
        		primary key (id)
        );
    • 3) 주의할 점(특징)

      • 엔티티 매니저 팩토리하나만 생성해서 애플리케이션 전체에서 공유
      • 엔티티 매니저는 쓰레드간에 공유X (사용하고 버려야 한다).
      • JPA의 모든 데이터 변경은 트랜잭션 안에서 실행
    • ❗4) 데이터 삽입

      package hellojpa;
      
      import javax.persistence.EntityManager;
      import javax.persistence.EntityManagerFactory;
      import javax.persistence.EntityTransaction;
      import javax.persistence.Persistence;
      
      public class JpaMain {
          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.setId(1L);
                  member.setName("HelloA");
      
                  em.persist(member);
      
                  tx.commit();
              } catch (Exception e){
                  tx.rollback();
              } finally {
                  em.close();
              }
              
              emf.close();
          }
      }
      • 💥애플리케이션 실행 시, 클래스를 찾지 못하는 오류가 발견 되면 아래와 같은 의존성을 주입하자.

        <dependency>
                <groupId>javax.xml.bind</groupId>
                <artifactId>jaxb-api</artifactId>
                <version>2.3.1</version>
            </dependency>
      • 삽입 결과

    • ❗5) 데이터 조회

      try {
              Member findMember = em.find(Member.class, 1L);
              System.out.println("findMember.id = " + findMember.getId()); //sql
              System.out.println("findMember.name = " + findMember.getName()); //영속성 컨텍스트
              
              tx.commit();
          }
      • 조회 결과

        //print:
        Hibernate: 
            select
                member0_.id as id1_0_0_,
                member0_.name as name2_0_0_ 
            from
                Member member0_ 
            where
                member0_.id=?
        findMember.id = 1
        findMember.name = HelloA
    • ❗6) 데이터 삭제

      try {
              Member findMember = em.find(Member.class, 1L);
      
              em.remove(findMember);
              tx.commit();
          }
    • ❗7) 데이터 수정

      try {
              Member findMember = em.find(Member.class, 1L);
              findMember.setName("HelloJPA");
      
              tx.commit();
          }
      • em.persist(findMemver)저장하는 문장 필요 x ⇒ Why? JPA를 통해 가져온 Entity는 JPA가 관리를 하고, 트랜잭션을 커밋하는 시점에 변경을 다 체크한다. (= 알아서 update 커밋) ⇒ 🧨변경 감지

      • 결과

  • JPQL

    • Java Persistence Query Language : 객체 지향 쿼리 (방언에 맞도록 변화한다.)

    • 전체 조회

      package hellojpa;
      
      import javax.persistence.EntityManager;
      import javax.persistence.EntityManagerFactory;
      import javax.persistence.EntityTransaction;
      import javax.persistence.Persistence;
      import java.util.List;
      
      public class JpaMain {
          public static void main(String[] args) {
              EntityManagerFactory emf = Persistence.createEntityManagerFactory("hello");
      
              EntityManager em = emf.createEntityManager();
      
              EntityTransaction tx = em.getTransaction();
              tx.begin();
      
              try {
                  List<Member> result = em.createQuery("select m from Member as m", Member.class).getResultList();
      
                  for (Member member : result) {
                      System.out.println("member.name = " + member.getName());
                  }
                  tx.commit();
              } catch (Exception e){
                  tx.rollback();
              } finally {
                  em.close();
              }
      
              emf.close();
          }
      }
      • 결과

        Hibernate: 
            /* select
                m 
            from
                Member as m */ select
                    member0_.id as id1_0_,
                    member0_.name as name2_0_ 
                from
                    Member member0_
        member.name = HelloJPA
        member.name = HelloB
      • 페이징 (offset, limit 설정)

        try {
                    List<Member> result = em.createQuery("select m from Member as m", Member.class)
                            .setFirstResult(5)
                            .setMaxResults(8)
                            .getResultList();
                    for (Member member : result) {
                        System.out.println("member.name = " + member.getName());
                    }
                    tx.commit();
                }
      • 조건 조회

        try {
                    List<Member> result = em.createQuery("select m from Member as m where m.id >= 2", Member.class)
                            .getResultList();
                    for (Member member : result) {
                        System.out.println("member.name = " + member.getName());
                    }
                    tx.commit();
                }
        • jpql문 안에 where를 사용
    • JPA를 사용하면 엔티티 객체 중심으로 개발, 문제는 필터링을 통한 검색 쿼리 ⇒ JPQL

    • JPQL ⇒ 엔티티 객체를 대상으로 쿼리

    • SQL ⇒ DB 테이블을 대상으로 쿼리

profile
열심히 해보자9999

0개의 댓글