java list 过滤数据库_java-使用spring-data-jpa和spring-m过滤数据库行

我有一个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 travels = travelRep.findByLastNameLike("%"+lastName+"%", pageable);

PageWrapper page = new PageWrapper(travels, "/search");

mav.addObject("page", page);

mav.addObject("lastName", lastName);

return mav;

}

效果很好:用户有一个带有Travel输入框的表格,该表格可用于过滤旅行记录。

除lastName之外,我的Travel域对象还具有许多我想用来过滤的属性。 我认为,如果这些属性都是字符串,那么我可以将它们添加为Likes,并添加spring-data-jpa方法以通过这些方法进行查询。 例如,我将添加方法TravelSearch。

但是,当需要过滤外键时,我不知道该怎么办。 因此,我的Travel具有Like属性,该属性是TravelSearch域对象的外键,我需要将其作为下拉菜单,以便用户选择Period。

我想做的是,当时间段为null时,我要检索由lastName过滤的所有旅行;当时间段不为null时,我要检索由lastName过滤的该时间段的所有旅行。

我知道如果在存储库中实现两个方法并对控制器使用Travel,则可以完成此操作:

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;

}

有没有一种方法可以不使用Travel? 我的旅行不仅具有期间,还具有需要使用下拉列表过滤的其他属性! 如您所知,当我需要使用更多下拉菜单时,复杂度将成倍增加,因为需要考虑所有组合:(

13年3月12日更新:接替Deinum先生的出色回答,在实际实施之后,我想提出一些评论,以确保问题/答案的完整性:

而不是实施Travel,应实现Like以避免类型检查警告。

请看看kostja对这个问题的出色回答真正动态的JPA CriteriaBuilder因为如果您要使用正确的过滤器,则需要实施此操作。

我可以找到的有关Criteria API的最佳文档是[http://www.ibm.com/developerworks/library/j-typesafejpa/。]这是一篇相当长的文章,但我完全推荐它-在阅读最多之后 我对Root和CriteriaBuilder的问题得到了回答:)

无法重用Travel对象,因为它包含我需要使用Like搜索的各种其他对象(也包含其他对象)-相反,我使用了包含我需要搜索的字段的TravelSearch对象。

2015年10月5日更新:按照@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 {

private TravelSearch criteria;

public TravelSpecification(TravelSearch ts) {

criteria= ts;

}

@Override

public Predicate toPredicate(Root root, CriteriaQuery> query,

CriteriaBuilder cb) {

Join o = root.join(Travel_.candidacy);

Path candidacy = root.get(Travel_.candidacy);

Path student = candidacy.get(Candidacy_.student);

Path lastName = student.get(Student_.lastName);

Path school = student.get(Student_.school);

Path period = candidacy.get(Candidacy_.period);

Path travelStatus = root.get(Travel_.travelStatus);

Path travelType = root.get(Travel_.travelType);

Path company = root.get(Travel_.company);

Path companyName = company.get(Company_.name);

final List predicates = new ArrayList();

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 travels = travelRep.findAll(tspec, pageable);

PageWrapper page = new PageWrapper(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;

}

希望我能帮上忙!

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值