스프링 DB 접근 기술
- H2 데이터베이스 설치
- 순수 JDBC
- 스프링 통합 테스트
- 스프링 JdbcTemplate
- JPA
- 스프링 데이터 JPA
create table member
(
id bigint generated by default as identity, -- null insert 되면 자동 채움
name varchar(255),
primary key (id) -- PK는 id
);
insert into member(name) values('spring');
insert into member(name) values('spring2');
insert into member(name) values('spring3');
select * from member;
dependencies {
implementation 'org.springframework.boot:spring-boot-starter-thymeleaf'
implementation 'org.springframework.boot:spring-boot-starter-web'
testImplementation 'org.springframework.boot:spring-boot-starter-test'
implementation 'org.springframework.boot:spring-boot-starter-jdbc'
runtimeOnly 'com.h2database:h2'
}
spring.datasource.url =jdbc:h2:tcp://localhost/~/test
spring.datasource.driver-class-name=org.h2.Driver
spring.datasource.username=sa
h2.Driver 또는 jdbc import 에러나면 build.gradle 가서 Load Gradle Changes로 import (코끼리아이콘)
hello.hellospring/repository/JdbcMemberRepository.java 생성
강의자료 p.38 'Jdbc 회원 리포지토리' 참고
import javax.sql.DataSource;
private final DataSource dataSource;
public SpringConfig(DataSource dataSource) {
this.dataSource = dataSource;
}
@Bean
public MemberRepository memberRepository() {
// return new MemoryMemberRepository();
return new JdbcMemberRepository(dataSource);
}
강의자료 p.45 '회원 서비스 스프링 통합 테스트' 참고
@SpringBootTest
: 스프링 컨테이너와 테스트를 함께 실행한다.@Transactional
: 테스트 케이스에 이 어노테이션이 있으면, 테스트 시작 전에 트랜잭션을 시작하고, 테스트 완료 후에 항상 롤백한다. 이렇게 하면 DB에 데이터가 남지 않으므로 다음 테스트에 영향을 주지 않는다.강의자료 p.47 '스프링 JdbcTemplate 회원 리포지토리' 참고
강의자료 p.49 참고
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'
}
}
spring-boot-starter-data-jpa
는 내부에 jdbc 관련 라이브러리를 포함한다. 따라서 jdbc는 제거해도 된다.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
를 추가해야 함
show-sql
: JPA가 생성하는 SQL을 출력한다.ddl-auto
: JPA는 테이블을 자동으로 생성하는 기능을 제공하는데 none
을 사용하면 해당 기능을 끈다.create
를 사용하면 엔티티 정보를 바탕으로 테이블도 직접 생성해준다.강의자료 p.51 참고
강의자료 p.52 참고
import org.springframework.transaction.annotation.Transactional
@Transactional
public class MemberService {}
org.springframework.transaction.annotation.Transactional
를 사용하자.강의자료 p.53 참고
주의) 스프링 데이터 JPA는 JPA를 편리하게 사용할 수 있도록 도와주는 기술이므로 JPA를 먼저 학습한 뒤, 스프링 데이터 JPA를 학습해야 함
package hello.hellospring.repository;
import hello.hellospring.domain.Member;
import org.springframework.data.jpa.repository.JpaRepository;
import java.util.Optional;
public interface SpringDataJpaMemberRepository extends JpaRepository<Member, Long>, MemberRepository {
Optional<Member> findByName(String name);
}
package hello.hellospring;
import hello.hellospring.repository.*;
import hello.hellospring.service.MemberService;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration
public class SpringConfig {
private final MemberRepository memberRepository;
public SpringConfig(MemberRepository memberRepository) {
this.memberRepository = memberRepository;
}
@Bean
public MemberService memberService() {
return new MemberService(memberRepository);
}
}
SpringDataJpaMemberRepository
를 스프링 빈으로 자동 등록해준다.findByName()
, findByEmail()
처럼 메서드 이름 만으로 조회 기능 제공참고) 실무에서는 JPA와 스프링 데이터 JPA를 기본으로 사용하고, 복잡한 동적 쿼리는 Querydsl이라는 라이브러리 사용하면 된다. Querydsl을 사용하면 쿼리도 자바코드로 안전하게 작성할 수 있고, 동적 쿼리도 편리하게 작성 가능하다. 이 조합으로 해결하기 어려운 쿼리는 JPA가 제공하는 네이티브 쿼리를 사용하거나, 앞서 학습한 JdbcTemplate을 사용하면 된다.