使用 spring-data-jpa 和 spring-mvc 过滤数据库行
我有一个spring-mvc项目,它使用spring-data-jpa进行数据访问。我有一个名为域对象,我希望允许最终用户对其应用许多筛选器。Travel
为此,我实现了以下控制器:
@Autowired
private TravelRepository travelRep;
@RequestMapping("/search")
public ModelAndView search(
@RequestParam(required= false, defaultValue="") String lastName,
Pageable pageable) {
ModelAndView mav = new ModelAndView("travels/list");
Page<Travel> travels = travelRep.findByLastNameLike("%"+lastName+"%", pageable);
PageWrapper<Travel> page = new PageWrapper<Travel>(travels, "/search");
mav.addObject("page", page);
mav.addObject("lastName", lastName);
return mav;
}
这工作正常:用户有一个带有输入框的表单,可用于过滤旅行。lastName
除了 lastName 之外,我的域对象还有很多属性可供我筛选。我认为,如果这些属性都是字符串,那么我可以将它们添加为s,并添加一个spring-data-jpa方法来查询这些属性。例如,我会添加一个方法。Travel
@RequestParam
findByLastNameLikeAndFirstNameLikeAndShipNameLike
但是,我不知道当我需要过滤外键时,我该怎么做。因此,我有一个属性是域对象的外键,我需要将其作为下拉列表,以便用户选择.Travel
period
Period
Period
我想做的是,当周期为空时,我想检索按姓氏过滤的所有行程,当句点不为空时,我想检索按姓氏过滤的此时间段的所有行程。
我知道,如果我在我的存储库中实现两个方法并使用一个到我的控制器,这可以完成:if
public ModelAndView search(
@RequestParam(required= false, defaultValue="") String lastName,
@RequestParam(required= false, defaultValue=null) Period period,
Pageable pageable) {
ModelAndView mav = new ModelAndView("travels/list");
Page travels = null;
if(period==null) {
travels = travelRep.findByLastNameLike("%"+lastName+"%", pageable);
} else {
travels = travelRep.findByPeriodAndLastNameLike(period,"%"+lastName+"%", pageable);
}
mav.addObject("page", page);
mav.addObject("period", period);
mav.addObject("lastName", lastName);
return mav;
}
有没有办法在不使用的情况下执行此操作?我的旅行不仅有句点,还有需要使用下拉列表过滤的其他属性!如您所知,当我需要使用更多下拉列表时,复杂性将呈指数级增加,因为所有组合都需要考虑:(if
03/12/13 更新:继续 M. Deinum 的精湛回答,在实际实现它之后,我想为问题/答案的完整性提供一些评论:
与其实现,不如实现以避免类型检查警告。
JpaSpecificationExecutor
JpaSpecificationExecutor<Travel>
请看一下kostja对这个问题的出色回答 真正动态的JPA CriteriaBuilder,因为如果你想拥有正确的过滤器,你需要实现这一点。
我能找到的关于标准API的最好的文档是 http://www.ibm.com/developerworks/library/j-typesafejpa/。这是一个相当长的阅读,但我完全推荐它 - 阅读后,我对Root和CriteriaBuilder的大多数问题都得到了:)
重用该对象是不可能的,因为它包含各种其他对象(也包含其他对象),我需要使用这些对象进行搜索 - 相反,我使用了一个包含我需要搜索的字段的对象。
Travel
Like
TravelSearch
2015 年 5 月 10 日更新:根据 @priyank 的要求,以下是我实现 TravelSearch 对象的方式:
public class TravelSearch {
private String lastName;
private School school;
private Period period;
private String companyName;
private TravelTypeEnum travelType;
private TravelStatusEnum travelStatus;
// Setters + Getters
}
这个对象是由TravelSpecification使用的(大多数代码是特定于域的,但我把它留在那里作为一个例子):
public class TravelSpecification implements Specification<Travel> {
private TravelSearch criteria;
public TravelSpecification(TravelSearch ts) {
criteria= ts;
}
@Override
public Predicate toPredicate(Root<Travel> root, CriteriaQuery<?> query,
CriteriaBuilder cb) {
Join<Travel, Candidacy> o = root.join(Travel_.candidacy);
Path<Candidacy> candidacy = root.get(Travel_.candidacy);
Path<Student> student = candidacy.get(Candidacy_.student);
Path<String> lastName = student.get(Student_.lastName);
Path<School> school = student.get(Student_.school);
Path<Period> period = candidacy.get(Candidacy_.period);
Path<TravelStatusEnum> travelStatus = root.get(Travel_.travelStatus);
Path<TravelTypeEnum> travelType = root.get(Travel_.travelType);
Path<Company> company = root.get(Travel_.company);
Path<String> companyName = company.get(Company_.name);
final List<Predicate> predicates = new ArrayList<Predicate>();
if(criteria.getSchool()!=null) {
predicates.add(cb.equal(school, criteria.getSchool()));
}
if(criteria.getCompanyName()!=null) {
predicates.add(cb.like(companyName, "%"+criteria.getCompanyName()+"%"));
}
if(criteria.getPeriod()!=null) {
predicates.add(cb.equal(period, criteria.getPeriod()));
}
if(criteria.getTravelStatus()!=null) {
predicates.add(cb.equal(travelStatus, criteria.getTravelStatus()));
}
if(criteria.getTravelType()!=null) {
predicates.add(cb.equal(travelType, criteria.getTravelType()));
}
if(criteria.getLastName()!=null ) {
predicates.add(cb.like(lastName, "%"+criteria.getLastName()+"%"));
}
return cb.and(predicates.toArray(new Predicate[predicates.size()]));
}
}
最后,这是我的搜索方法:
@RequestMapping("/search")
public ModelAndView search(
@ModelAttribute TravelSearch travelSearch,
Pageable pageable) {
ModelAndView mav = new ModelAndView("travels/list");
TravelSpecification tspec = new TravelSpecification(travelSearch);
Page<Travel> travels = travelRep.findAll(tspec, pageable);
PageWrapper<Travel> page = new PageWrapper<Travel>(travels, "/search");
mav.addObject(travelSearch);
mav.addObject("page", page);
mav.addObject("schools", schoolRep.findAll() );
mav.addObject("periods", periodRep.findAll() );
mav.addObject("travelTypes", TravelTypeEnum.values());
mav.addObject("travelStatuses", TravelStatusEnum.values());
return mav;
}
希望我有帮助!