Spring Данные JPA: запрос по примеру?
Используя Spring Data JPA, я могу сделать запрос на примере, где в качестве критериев поиска используется конкретный экземпляр объекта?
Например (без каламбура), если у меня есть объект Person
, который выглядит следующим образом:
@Entity
public class Person {
private String firstName;
private String lastName;
private boolean employed;
private LocalDate dob;
...
}
Я мог найти всех занятых лиц с фамилией Смита, родившихся 1 января 1977 года, с примером:
Person example = new Person();
example.setEmployed(true);
example.setLastName("Smith");
example.setDob(LocalDate.of(1977, Month.JANUARY, 1));
List<Person> foundPersons = personRepository.findByExample(example);
Ответы
Ответ 1
Spring данные зависят от JPA и EntityManager, а не от Hibernate и Session, и поэтому у вас нет findByExample из коробки. Вы можете использовать создание автоматического запроса данных spring и написать метод в вашем репозитории со следующей подписью:
List<Person> findByEmployedAndLastNameAndDob(boolean employed, String lastName, LocalDate dob);
Ответ 2
Теперь это возможно с помощью Spring Data. Проверьте http://docs.spring.io/spring-data/jpa/docs/current/reference/html/#query-by-example
Person person = new Person();
person.setLastname("Smith");
Example<Person> example = Example.of(person);
List<Person> results = personRepository.findAll(example);
Обратите внимание, что для этого требуются очень свежие версии 2016
<dependency>
<groupId>org.springframework.data</groupId>
<artifactId>spring-data-jpa</artifactId>
<version>1.10.1.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework.data</groupId>
<artifactId>spring-data-commons</artifactId>
<version>1.12.1.RELEASE</version>
</dependency>
см. https://github.com/paulvi/com.example.spring.findbyexample
Ответ 3
Используя интерфейс Spring data Specification
, я смог аппроксимировать использование запроса на примере. Здесь класс PersonSpec
, который реализует Specification
и требует "примерного" человека для настройки Predicate
, возвращаемого Specification
:
public class PersonSpec implements Specification<Person> {
private final Person example;
public PersonSpec(Person example) {
this.example = example;
}
@Override
public Predicate toPredicate(Root<Person> root, CriteriaQuery<?> cq, CriteriaBuilder cb) {
List<Predicate> predicates = new ArrayList<>();
if (StringUtils.isNotBlank(example.getLastName())) {
predicates.add(cb.like(cb.lower(root.get(Person_.lastName)), example.getLastName().toLowerCase() + "%"));
}
if (StringUtils.isNotBlank(example.getFirstName())) {
predicates.add(cb.like(cb.lower(root.get(Person_.firstName)), example.getFirstName().toLowerCase() + "%"));
}
if (example.getEmployed() != null) {
predicates.add(cb.equal(root.get(Person_.employed), example.getEmployed()));
}
if (example.getDob() != null) {
predicates.add(cb.equal(root.get(Person_.dob), example.getDob()));
}
return andTogether(predicates, cb);
}
private Predicate andTogether(List<Predicate> predicates, CriteriaBuilder cb) {
return cb.and(predicates.toArray(new Predicate[0]));
}
}
Репозиторий просто:
import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.data.jpa.repository.JpaSpecificationExecutor;
public interface PersonRepository extends JpaRepository<Person, Long>, JpaSpecificationExecutor {}
Пример использования:
Person example = new Person();
example.setLastName("James");
example.setEmployed(true);
PersonSpec personSpec = new PersonSpec(example);
List<Person> persons = personRepository.findAll(personSpec);