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 해주기!
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가 꼭 필요하다 !
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
에 다음과 같이 추가
transactional
은org.springframework.transaction.annotation.Transactional
을 사용- 스프링은 해당 클래스의 메서드를 실행할 때 트랜잭션을 시작하고, 메서드가 정상 종료되면 트랜잭션을 커밋함
- 만약 런타임 예외가 발생하면 롤백
- 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 라는 프레임워크를 더한다면, 리포지토리에 구현 클래스 없이 인터페이스 만으로 개발을 할 수 있다!!!!
또한 기본 CRUD 기능도 스프링 데이터 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
를 사용한다. 그리고 인터페이스는 다중 상속이 가능하다.
+) 자 이제 구현해보자...! 하려고 했는데 이미 구현이 다 끝나서 필요가 없다고 하셨다 ㅎ 정말 이 한 줄이 끝인 건가 참 신기했다😂
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가 제공하는 편리한 기능
findByName()
, findByEmail()
처럼 메서드 이름 만으로 조회 기능 제공 위의 인터페이스를 생성하는 과정에서 findByName
만 작성한 이유는 ❓
서버 개발자들은 이 모든 걸 두루두루 잘 알아야 한다는데... 그냥....내 머리속에 두루두루 섞여 있는 거 같다^_^
입문 강의를 마치면, 지금까지 배운 것들을 나름대로 다시 정리하는 포스팅을 해봐야겠다🌟