https://sundaland.tistory.com/400
[ ▶ Core concepts ]
[ ▷ Repository 인터페이스 ]
- 중심 인터페이스: Spring Data 리포지토리 추상화의 중심 인터페이스는 Repository이다. 이 인터페이스는 관리할 도메인 클래스와 도메인 클래스의 identifier 타입을 제네릭 타입 파라미터로 받는다. 기본적으로 이 인터페이스는 마커 인터페이스로 사용되며, 작업할 타입을 캡처하고 이를 확장하는 인터페이스를 발견하는 데 도움을 준다.
- 도메인 객체: Spring Data는 도메인 타입을 엔티티로 간주하며, 보다 구체적으로 집합체(aggregate)로 설명한다. 따라서 문서 전반에 걸쳐 "엔티티"라는 용어가 사용되며, 이는 "도메인 타입"이나 "집합체"와 상호 교환할 수 있다.
- 도메인 주도 설계(DDD): 도메인 객체는 DDD의 관점에서 정의됩니다. 도메인 객체는 identifier를 가지고 있으며, 특정 패턴으로 데이터에 접근할 때 이 identifier를 참조해야 한다. identifier를 참조하는 것은 리포지토리 및 쿼리 메서드를 다룰 때 더 의미 있게 된다.
[ ▷ CrudRepository 인터페이스 ]
CrudRepository는 관리되는 엔티티 클래스에 대한 고급 CRUD 기능을 제공한다.
public interface CrudRepository<T, ID> extends Repository<T, ID> {
<S extends T> S save(S entity); // (1)
Optional<T> findById(ID primaryKey); // (2)
Iterable<T> findAll(); // (3)
long count(); // (4)
void delete(T entity); // (5)
boolean existsById(ID primaryKey); // (6)
// ... 더 많은 기능 생략됨.
}
- save(S entity): 주어진 엔티티를 저장한다. (새로운 엔티티는 삽입되고, 기존 엔티티는 업데이트됨)
- findById(ID primaryKey): 주어진 ID로 식별되는 엔티티를 리턴한다.
- findAll(): 모든 엔티티를 리턴한다.
- count(): 현재 저장된 엔티티의 수를 리턴한다.
- delete(T entity): 주어진 엔티티를 삭제한다.
- existsById(ID primaryKey): 주어진 ID를 가진 엔티티가 존재하는지 여부를 나타낸다.
△ CRUD 메서드
CrudRepository 인터페이스에 선언된 메서드는 일반적으로 CRUD 메서드라고 불린다.
△ ListCrudRepository
ListCrudRepository는 CrudRepository와 동일한 메서드를 제공하지만, 리턴 타입이 List인 반면 CrudRepository는 Iterable을 리턴한다.
△ 주의사항
- 리포지토리 인터페이스는 findById(ID identifier)와 같은 몇 가지 예약된 메서드를 포함한다. 이 메서드는 도메인 타입의 identifier 속성을 목표로 하며, 속성 이름과 관계없이 동작한다.
- 만약 식별자를 참조하는 속성이 Id라는 이름이 아닐 경우, @Query 애노테이션을 사용하여 커스텀 쿼리를 제공할 수 있다. 그러나 이렇게 하면 혼란스러울 수 있으며, ID 타입과 Id 속성의 타입이 다를 경우에는 더 많은 제약을 받을 수 있으므로 이러한 경로를 피하는 것이 좋다.
[ ▷ JpaRepository 및 MongoRepository ]
JpaRepository 또는 MongoRepository와 같은 특정 영속성 기술에 대한 추상화도 제공된다. 이러한 인터페이스는 CrudRepository를 확장하며, 기본적인 CRUD 기능 외에 특정 영속 기술의 기능을 추가로 노출한다.
[ ▷ PagingAndSortingRepository 인터페이스 ]
PagingAndSortingRepository와 ListPagingAndSortingRepository는 엔티티에 대한 페이지 접근을 쉽게 하기 위해 추가적인 메서드를 제공한다.
public interface PagingAndSortingRepository<T, ID> {
Iterable<T> findAll(Sort sort);
Page<T> findAll(Pageable pageable);
}
- findAll(Sort sort): 주어진 정렬 기준에 따라 모든 엔티티를 리턴한다.
- findAll(Pageable pageable): 주어진 페이지 요청에 따라 엔티티를 페이지 단위로 리턴한다.
△ 주의사항
확장 인터페이스는 실제 저장소 모듈에 의해 지원된다. 따라서 이 문서에서 설명하는 일반적인 체계와는 별개로, 사용하려는 인터페이스가 사용 중인 저장소 모듈에서 지원되는지 확인해야 한다.
△ 예제: 페이지 접근
사용자(User) 엔티티에 대해 페이지 크기가 20인 두 번째 페이지에 접근하려면 다음과 같이 할 수 있다.
PagingAndSortingRepository<User, Long> repository = // ... bean에 접근
Page<User> users = repository.findAll(PageRequest.of(1, 20));
ListPagingAndSortingRepository는 동일한 메서드를 제공하지만, 리턴 타입이 List인 반면 PagingAndSortingRepository는 Iterable을 리턴한다.
[ ▷ 쿼리 메서드 ]
쿼리 메서드에 대한 파생 쿼리도 가능하다. 다음은 파생된 카운트 쿼리의 인터페이스 정의이다.
△ 파생된 카운트 쿼리
interface UserRepository extends CrudRepository<User, Long> {
long countByLastname(String lastname);
}
△ 파생된 삭제 쿼리
interface UserRepository extends CrudRepository<User, Long> {
long deleteByLastname(String lastname);
List<User> removeByLastname(String lastname);
}
- countByLastname(String lastname): 주어진 성(lastname)을 가진 사용자 수를 리턴한다.
- deleteByLastname(String lastname): 주어진 성을 가진 사용자를 삭제하고 삭제된 사용자 수를 리턴한다.
- removeByLastname(String lastname): 주어진 성을 가진 사용자를 삭제하고 삭제된 사용자 목록을 리턴한다.
Spring Data 리포지토리의 기본 구조와 사용법을 이해하는 것은 애플리케이션 개발 시 데이터에 접근하고 조작하는 데 큰 도움이 된다. CrudRepository, PagingAndSortingRepository, 그리고 이들 인터페이스의 파생된 메서드들에 대한 이해는 데이터 처리의 유연성과 효율성을 높여줄 것이다. 이러한 기능들은 데이터베이스와의 상호작용을 단순화하고, 도메인 객체에 대한 CRUD 작업을 손쉽게 수행할 수 있게 도와준다.
'Spring Boot > Spring Data JPA' 카테고리의 다른 글
Defining Query Methods (0) | 2024.10.22 |
---|---|
Persisting Entities (0) | 2024.10.22 |
Configuration (0) | 2024.10.22 |
Defining Repository Interfaces (0) | 2024.10.22 |
Getting Started (0) | 2024.10.22 |