프로그래밍/Spring

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

daykim 2022. 10. 26. 16:14
김영한 스프링 입문 강의 정리
 

[무료] 스프링 입문 - 코드로 배우는 스프링 부트, 웹 MVC, DB 접근 기술 - 인프런 | 강의

스프링 입문자가 예제를 만들어가면서 스프링 웹 애플리케이션 개발 전반을 빠르게 학습할 수 있습니다., - 강의 소개 | 인프런...

www.inflearn.com

목차

  • H2 DB 설치
  • 순수 JDBC
  • 스프링 통합 테스트

 

H2 데이터베이스 설치

H2는 자바로 작성된 관계형 데이터베이스 관리 시스템

개발이나 테스트 용도로 가볍고 편리한 DB, 웹 화면 제공

왼쪽 상단은 나가기 버튼

테이블 생성하기

DROP TABLE IF EXISTS MEMBER CASCADE;
CREATE TABLE MEMBER
(
    id bigint generated by default as identity,
    name varchar(255),
    primary key(id)
);
  • generated by default as identity
    null값이 들어왔을 때 자동으로 id 값을 채워준다.

왼쪽의 Member 클릭 시 자동 생성됨

데이터 넣기

INSERT INTO MEMBER(NAME) VALUES('spring')

 

순수 JDBC

환경 설정

build.gradle 파일에 jdbc, h2 데이터베이스 관련 라이브러리 추가

implementation 'org.springframework.boot:spring-boot-starter-jdbc'
runtimeOnly 'com.h2database:h2'

스프링 부트 데이터베이스 연결 설정 추가

// resources/application.properties
spring.datasource.url=jdbc:h2:tcp://localhost/~/test
spring.datasource.driver-class-name=org.h2.Driver
spring.datasource.username=sa

JDBC 리포지토리 구현

주의
이렇게 JDBC API로 직접 코딩하는 것은 15년 전 이야기
따라서 고대 개발자들이 이렇게 고생하고 살았구나 생각하고, 정신건강 위해서 참고만 해라.

JDBC 회원 리포지토리

스프링 설정 변경

package hello.hellospring;

import hello.hellospring.repository.JdbcMemberRepository;
import hello.hellospring.repository.MemberRepository;
import hello.hellospring.repository.MemoryMemberRepository;
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.sql.DataSource;

@Configuration
public class SpringConfig {

    private DataSource dataSource;

    @Autowired
    public SpringConfig(DataSource dataSource) {
        this.dataSource = dataSource;
    }

    @Bean
    public MemberService memberService(){

        return new MemberService(memberRepository());
    }

    @Bean
    public MemberRepository memberRepository(){

        // return new MemoryMemberRepository();
        return new JdbcMemberRepository(dataSource);
    }
}

 

h2 데이터베이스 데이터

DataSource는 데이터베이스 커넥션을 획득할 때 사용하는 객체다.
스프링 부트는 데이터베이스 커넥션 정보를 바탕으로 DataSource를 생성하고, 스프링 빈으로 만들어둔다.
그래서 DI를 받을 수 있다.

객체지향 - 다형성 활용
Interface를 두고 구현체를 바꿔끼우기 가능했다.
MemoryMemberRepository() -> JdbcMemberRepository()

구현 클래스 추가 이미지

스프링 설정 이미지

개방 · 폐쇄 원칙(OCP, Open-Closed Principle)

  • 확장에는 열려있고, 수정에는 닫혀있다.
  • 객체지향 다형성 개념을 잘 활용하면 가능

스프링의 DI를 사용하면 기존 코드를 전혀 손대지 않고, 설정만으로 클래스를 변경할 수 있다.

데이터를 DB에 저장하므로 스프링 서버를 다시 실행해도 데이터가 안전하게 저장된다.

 

스프링 통합 테스트

스프링 컨테이너와 DB까지 연결한 통합 테스트를 진행해보자.

회원 서비스 스프링 통합 테스트

// MemberServiceIntegrationTest
package hello.hellospring.service;

import hello.hellospring.domain.Member;
import hello.hellospring.repository.MemberRepository;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.transaction.annotation.Transactional;

import static org.assertj.core.api.Assertions.assertThat;
import static org.junit.jupiter.api.Assertions.assertThrows;

@SpringBootTest
@Transactional
public class MemberServiceIntegrationTest {

    @Autowired MemberService memberService;
    @Autowired MemberRepository memberRepository;

    @Test
    void 회원가입() {
        // given : 무언가 주어졌을 때
        Member member = new Member();
        member.setName("spring");

        // when : 이것을 실행했을 때
        Long saveId = memberService.join(member);

        // then : 결과가 이래야 한다.
        Member findMember = memberService.findOne(saveId).get();
        assertThat(member.getName()).isEqualTo(findMember.getName());
    }

    @Test
    public void 중복_회원_예외() {
        // given
        Member member1 = new Member();
        member1.setName("spring");

        Member member2 = new Member();
        member2.setName("spring");

        // when
        memberService.join(member1);
        IllegalStateException e = assertThrows(IllegalStateException.class, () -> memberService.join(member2));

        assertThat(e.getMessage()).isEqualTo("이미 존재하는 회원입니다.");

    }
}

@SpringBootTest

스프링 컨테이너와 테스트를 함께 실행한다.

@Transactional

테스트 케이스에 어노테이션이 있으면, 테스트 시작 전에 트랜잭션을 시작하고, 테스트 완료 후에 항상 롤백한다.
이렇게 하면 DB에 데이터가 남지 않으므로 다음 테스트에 영향을 주지 않는다.