Spring/스프링 입문 - 코드로 배우는 스프링 부트, 웹 MVC, DB 접근 기술

[ 스프링 DB 접근 기술 ] JPA

alsruds 2023. 9. 16. 04:20

🙂강의🙂

https://www.inflearn.com/course/%EC%8A%A4%ED%94%84%EB%A7%81-%EC%9E%85%EB%AC%B8-%EC%8A%A4%ED%94%84%EB%A7%81%EB%B6%80%ED%8A%B8

 

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

스프링 입문자가 예제를 만들어가면서 스프링 웹 애플리케이션 개발 전반을 빠르게 학습할 수 있습니다., 스프링 학습 첫 길잡이! 개발 공부의 길을 잃지 않도록 도와드립니다. 📣 확인해주세

www.inflearn.com

 

💡현재 해결되지 않은 것💡 : SQL 은 개발자가 직접 작성 ➡️ JPA 사용 시 해결 가능

✅ JPA 는 기존의 반복 코드 & 기본적인 SQL 처리, 실행
✅ JPA 사용 시 SQL 과 데이터 중심의 설계에서 객체 중심의 설계로 전환 가능
✅ JPA 사용 시 개발 생산성 향상

 

[ JPA, H2 데이터베이스 관련 라이브러리 추가 ]

  • build.gradle 파일 수정하기 ➡️ 수정 후 refresh 필수 !
...

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'
}

...

→ spring-boot-starter-data-jpa 가 jdbc 관련 라이브러리를 포함하고 있기 때문에 jdbc 는 제거 가능

 

[ 스프링 부트에 JPA 설정 추가 ]

  • resources/application.properties 파일 수정하기
...
spring.jpa.show-sql=true
spring.jpa.hibernate.ddl-auto=none

show-sql : JPA 가 생성하는 SQL 출력

ddl-auto : JPA 가 제공하는 자동 테이블 생성 기능

          none : 기능 제거

          create : 엔티티 정보를 바탕으로 테이블 생성

 

[ JPA 엔티티 매핑 ]

  • src/main/java/hello.hellospring/domain/Member 설정 변경하기
package hello.hellospring.domain;

import jakarta.persistence.*;

@Entity // JPA 가 관리하는 엔티티라는 뜻
public class Member {

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY) // DB 가 알아서 ID 생성
    private Long id;
    
    @Column // (name = "username") : DB 에 있는 column 명이 username 이 됨
    private String name;

    ...
}

 

[ JPA 회원 리포지토리 생성 ]

  • src/main/java/hello.hellospring/repository 에 JpaMemberRepository 생성하기
package hello.hellospring.repository;

import hello.hellospring.domain.Member;
import jakarta.persistence.EntityManager;
import org.springframework.transaction.annotation.Transactional;

import java.util.List;
import java.util.Optional;

public class JpaMemberRepository implements MemberRepository{

    private final EntityManager em;

    public JpaMemberRepository(EntityManager em) {
        this.em = em;
    }

    @Override
    public Member save(Member member) {
        em.persist(member);
        return member;
    }

    @Override
    public Optional<Member> findById(Long id) {
        Member member = em.find(Member.class, id);
        return Optional.ofNullable(member);
    }

    @Override
    public Optional<Member> findByName(String name) {
        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();
    }
}

 

[ 서비스 계층에 트랜잭션 추가 ]

  • src/main/java/hello.hellospring/service/MemberService 파일 수정하기
...

@Transactional
public class MemberService {

    ...
}

→ JPA 를 통한 모든 데이터 변경은 트랜잭션 안에서 실행해야 함

 

[ 스프링 설정 변경 ]

  • SpringConfig 파일 수정하기
...

@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 JpaMemberRepository(em);
    }
}

 

  • 테스트 하기

성공 !