[Spring Boot][1] 6-3. 스프링 DB 접근 기술

🏷 JPA

  • JPA는 기존의 반복 코드는 물론이고, 기본적인 SQL도 JPA가 직접 만들어서 실행해준다! (개발자가 직접 작성할 필요가 없다)
  • JPA를 사용하면, SQL과 데이터 중심의 설계에서 객체 중심의 설계로 패러다임을 전환을 할 수 있다.
  • JPA를 사용하면 개발 생산성을 크게 높일 수 있다

✔️ build.gradle 에 라이브러리 추가하기

dependencies {
      implementation 'org.springframework.boot:spring-boot-starter-thymeleaf'
      implementation 'org.springframework.boot:spring-boot-starter-web'
//implementation 'org.springframework.boot:spring-boot-starter-jdbc' implementation 'org.springframework.boot:spring-boot-starter-data-jpa' runtimeOnly 'com.h2database:h2' testImplementation('org.springframework.boot:spring-boot-starter-test') {
          exclude group: 'org.junit.vintage', module: 'junit-vintage-engine'
      }
}

코드 추가해주고 코끼리 이모티콘 눌러서 refresh 해주기!


✔️ JPA와 관련된 설정들도 추가해주기

main/resources/application.properties

spring.datasource.url=jdbc:h2:tcp://localhost/~/test
spring.datasource.driver-class-name=org.h2.Driver
spring.datasource.username=sa
// 여기서부터 추가
spring.jpa.show-sql=true
spring.jpa.hibernate.ddl-auto=none

🤚🏻 주의!: 스프링부트 2.4부터는 spring.datasource.username=sa꼭 추가해주어야 한다. 그렇지 않으면 오류가 발생한다.


✔️ JPA 엔티티 매핑하기

JPA 란 ❓
: 인터페이스이다 !
: 객체와 ORM이라는 기술 !

main/java/hello.hellospring/domain/Member.java

package hello.hellospring.domain;

import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;

// @Entity : JPA가 관리하는 entity
@Entity
public class Member {

    // IDENTITY : 디비가 알아서 생성해주는 방법
    @Id @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;    // 우리가 정하는 아이디가 아닌 시스템 자체에서 생성하는 아이디 (구별하기 위함)
    private String name;

    // 만약 DB에 우리는 똑같이 name이라 되어있지만 username이다? 그러면
    // @Column(name="username") 이렇게 해주면 mapping 됨.
    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;
    }
}

💡 JPA를 사용하기 위해서는 EntityManager가 꼭 필요하다 !


✔️ JPA 회원 리포지토리 만들기

main/java/hello.hellospring/repository/JpaMemberRepository.java

package hello.hellospring.repository;

import hello.hellospring.domain.Member;
import javax.persistence.EntityManager;
import java.util.List;
import java.util.Optional;

public class JpaMemberRepository implements MemberRepository {

    // Jpa는 EntitiyManager로 모든게 동작함
    private final EntityManager em;
    public JpaMemberRepository(EntityManager em) {
        this.em = em;
    }

    @Override
    public Member save(Member member) {
        // persist : 영구저장하다
        em.persist(member);
        return member;
    }

    @Override
    public Optional<Member> findById(Long id) {
        // 조회 (클래스와 pk값)
        Member member = em.find(Member.class, id);
        return Optional.ofNullable(member);
    }

    @Override
    public Optional<Member> findByName(String name) {
        // 객체지향쿼리 사용(=createQuery)
        // 보통 테이블을 대상으로 sql을 날리는데 객체를 대상으로 쿼리를 날림!
        // 그러면 이게 sql로 번역이 됨
        List<Member> result = em.createQuery("select m from Member m where m.name = :name", Member.class)
                .setParameter("name", name)
                .getResultList();
        return result.stream().findAny();
    }


    @Override
    public List<Member> findAll() {

        return em.createQuery("select m from Member m", Member.class)
                .getResultList();
    }
}

💡 JPA를 사용하기 위해서는 @Transactional 이 꼭 있어야 한다! 데이터를 저장하거나 변경할 때 필요하기 따문이다.
main/java/hello.hellospring/service/MemberService.java에 다음과 같이 추가

  • transactionalorg.springframework.transaction.annotation.Transactional 을 사용
  • 스프링은 해당 클래스의 메서드를 실행할 때 트랜잭션을 시작하고, 메서드가 정상 종료되면 트랜잭션을 커밋
  • 만약 런타임 예외가 발생하면 롤백
  • JPA를 통한 모든 데이터 변경은 트랜잭션 안에서 실행해야 한다.

✔️ JPA를 사용하도록 스프링 설정 변경

main/java/hello.hellospring/SpringConfig.java 파일 변경

package hello.hellospring;

import hello.hellospring.repository.*;
import hello.hellospring.service.MemberService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

import javax.persistence.EntityManager;
import javax.sql.DataSource;

@Configuration
public class SpringConfig {
    // 추가
    private EntityManager em;
    
    @Autowired
    public SpringConfig(EntityManager em) {
        this.em = em;
    }
   
    @Bean
    public MemberService memberService() {
        return new MemberService(memberRepository());
    }
    @Bean
    public MemberRepository memberRepository() {
//        return new MemoryMemberRepository();
//        return new JdbcMemberRepository(dataSource);
//        return new JdbcTemplateMemberRepository(dataSource);
        // 추가
        return new JpaMemberRepository(em);
    }
}

✔️ 잘 돌아가는지 확인

통합테스트 MemberServiceIntegrationTest.java 실행!

  • 회원가입 부분만 돌려봤을 때, 잘 돌아가는 것을 확인
  • 회원가입 위에 @Commit 어노테이션을 추가하면, 디비에 반영된다


🏷 스프링 데이터 JPA

지금까지 해 온 것처럼, 스프링부트와 JPA만 사용해도 개발 생산성이 매우 증가하고 편리한데, 이 기반 위에 스프링 데이터 JPA 라는 프레임워크를 더한다면, 리포지토리에 구현 클래스 없이 인터페이스 만으로 개발을 할 수 있다!!!!
또한 기본 CRUD 기능도 스프링 데이터 JPA가 모두 제공하여 개발 생산성을 높여주고, 중복을 확 줄여준다.
따라서, 실무에서 관계형 데이터베이스를 사용한다면 스프링 데이터 JPA나 JPA 관련된 기술은 필수 사항이 되었다 😃

여기서 주의할 점 🤚🏻
스프링 데이터 JPA는 JPA를 편리하게 사용하도록 도와주는 기술이므로, JPA를 먼저 학습한 후에 스프링 데이터 JPA를 학습해야 한다.

✔️ 스프링 데이터 JPA 회원 리포지토리 생성하기

main/java/hello.hellospring/repository/SpringDataJpaMemberRepository.java 파일 생성하기
(Class가 아닌 Repository 로 설정하고 생성!)

package hello.hellospring.repository;

import hello.hellospring.domain.Member;
import org.springframework.data.jpa.repository.JpaRepository;

import java.util.Optional;

// 인터페이스가 인터페이스를 상속할 땐 extends 사용, 인터페이스는 다중상속 가능
public interface SpringDataJpaMemberRepository extends JpaRepository<Member, Long>, MemberRepository {
    @Override
    Optional<Member> findByName(String name);
}

💡 인터페이스가 인터페이스를 상속할 땐 extends를 사용한다. 그리고 인터페이스는 다중 상속이 가능하다.
+) 자 이제 구현해보자...! 하려고 했는데 이미 구현이 다 끝나서 필요가 없다고 하셨다 ㅎ 정말 이 한 줄이 끝인 건가 참 신기했다😂


✔️ 스프링 데이터 JPA 회원 리포지토리를 사용하도록 스프링 설정 변경하기

main/java/hello.hellospring/SpringConfig.java 파일 변경

package hello.hellospring;

import hello.hellospring.repository.*;
import hello.hellospring.service.MemberService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class SpringConfig {
    private final MemberRepository memberRepository;

    // 생성자가 하나일 때는 생략 가능함
    @Autowired
    public SpringConfig(MemberRepository memberRepository) {
        this.memberRepository = memberRepository;
    }

    @Bean
    public MemberService memberService() {
        return new MemberService(memberRepository);
    }

//    @Bean
//    public MemberRepository memberRepository() {
//        return new MemoryMemberRepository();
//        return new JdbcMemberRepository(dataSource);
//        return new JdbcTemplateMemberRepository(dataSource);
        // 추가
//        return new JpaMemberRepository(em);
//    }

}

방금 전에 만든 SpringDataJpaMemberRepository 와 같이, 인터페이스만 만들어 놓으면(스프링 데이터가 제공하는 JpaRepository를 extends해서),
스프링 데이터 JPA 가 인터페이스에 대한 구현체를 자동으로 만들어주고 스프링 빈에도 등록해 준다!


✔️ 테스트해보기 !
통합 테스트를 진행하면, 위와 같이 초록불이 들어온 것을 확인할 수 있다.


✔️ 스프링 데이터 JPA 제공 클래스


💡스프링 데이터 JPA가 제공하는 편리한 기능

  • 인터페이스를 통한 기본적인 CRUD
  • findByName() , findByEmail() 처럼 메서드 이름 만으로 조회 기능 제공
  • 페이징 기능 자동 제공

위의 인터페이스를 생성하는 과정에서 findByName만 작성한 이유는 ❓

  • 이름으로 찾는 경우, 비즈니스가 다 다르기 때문에 공통화 하는것이 불가능함

✔️ 정리

  • JdbcTemplate : 반복되는 쿼리는 확실히 줄어들지만, SQL은 내가 직접 작성해야 함
  • JPA : 기본적인 CRUD를 다루는데 내가 직접 쿼리를 작성할 필요는 없었음. 물론 SELEC 할 때는 JPQL을 짜야 함.
  • 스프링 데이터 JPA : 아예 내가 구현 클래스를 작성할 필요 없이 인터페이스만으로 구현이 끝남! 매우 간편😃


서버 개발자들은 이 모든 걸 두루두루 잘 알아야 한다는데... 그냥....내 머리속에 두루두루 섞여 있는 거 같다^_^
입문 강의를 마치면, 지금까지 배운 것들을 나름대로 다시 정리하는 포스팅을 해봐야겠다🌟

좋은 웹페이지 즐겨찾기