如何将“过滤器”对象传递给Spring Data JPA存储库以创建查询

时间:2018-08-30 17:13:46

标签: java spring spring-data-jpa

我需要将一个对象传递给spring引导端点(在@RequestBody中),并使用该对象通过spring数据jpa查询数据库。

我的班级

@Entity(name = "PESSOA")
public class Pessoa {

private Long codigo;
private String nome;
private String cpf;
private String estadoCivil;
private LocalDate dataNascimento;

@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
public Long getCodigo() {
    return codigo;
}

@Column
public String getNome() {
    return nome;
}

@Column
public String getCpf() {
    return cpf;
}

@Column
public String getEstadoCivil() {
    return estadoCivil;
}

@Column
public LocalDate getDataNascimento() {
    return dataNascimento;
}

public void setCpf(String cpf) {
    this.cpf = cpf;
}

public void setEstadoCivil(String estadoCivil) {
    this.estadoCivil = estadoCivil;
}

public void setDataNascimento(LocalDate dataNascimento) {
    this.dataNascimento = dataNascimento;
}

public void setCodigo(Long codigo) {
    this.codigo = codigo;
}

public void setNome(String nome) {
    this.nome = nome;
}

}

这是包含我的端点的类:

@RestController
public class PessoaService {

@Autowired
private PessoaRepository pessoaRepository;

@GetMapping("/pessoa")
public List<Pessoa> all() {
    return pessoaRepository.findAll();
}


@GetMapping("/pessoa/{id}")
public Pessoa get(@PathVariable Long id) {
    return pessoaRepository.findById(id).get();
}

@PostMapping("/pessoa")
public Pessoa create(@RequestBody Pessoa pessoa) {
    return pessoaRepository.saveAndFlush(pessoa);
}

@PutMapping("/pessoa/{id}")
public Pessoa update(@PathVariable Long id, @RequestBody Pessoa pessoa) {
    Pessoa pessoaExistente = pessoaRepository.findById(id).get();
    BeanUtils.copyProperties(pessoa, pessoaExistente);
    return pessoaRepository.saveAndFlush(pessoaExistente);
}

@DeleteMapping("/pessoa/{id}")
public Pessoa delete(@PathVariable Long id) {
    Pessoa pessoaExistente = pessoaRepository.findById(id).get();
    pessoaRepository.delete(pessoaExistente);
    return pessoaExistente;
}

}

这是我的存储库:

@Repository
public interface PessoaRepository extends JpaRepository<Pessoa, Long> {

}

有一种创建类似的东西的方法:

@GetMapping("/pessoa")
public List<Pessoa> filter(@RequestBody PessoaFilter filter) {
    return pessoaRepository.findByFilter(filter);
}

Spring数据提供了可以用作过滤器的东西吗?

1 个答案:

答案 0 :(得分:2)

您可以使用规范来做到这一点。参见this tutorial on Spring's blog

一旦您向存储库中添加了接受该方法的过滤版本的接口,您正在调用的PessoaFilter便会作为谓词传递到查询方法中。

实现此界面:

public interface Specification<T> {
    Predicate toPredicate(Root<T> root, CriteriaQuery query, CriteriaBuilder cb);
}

可以在此处实现,例如:

public interface PessoaSpecifications {

    static <Pessoa> Specification<Pessoa> byCpf(String someCpf) {
        return new Specification<Pessoa> {
            public Predicate toPredicate(Root<Pessoa> root, CriteriaQuery query, CriteriaBuilder cb) {
                return cb.equal(root.get(Customer_.cpf), someCpf);
            }
        };
    }
}

并使您的存储库除之前扩展的内容外扩展JpaSpecificationExecutor。您将可以将规范传递给查询:

pessoaRepo.findAll(PessoaSpecifications.byCpf("asdf"));

同一篇文章继续描述了使用Querydsl进行此操作的另一种方法,该方法避免了创建生成谓词的代码的麻烦。将querydsl插件添加到pom.xml并使用QueryDslPredicateExecutor扩展存储库接口后,您可以编写类似

的内容
List<Pessoa> p = pessoaRepo.findAll(pessoa.cpf.eq("asdf"));