当前位置: 首页 > 知识库问答 >
问题:

用spring-data-jpa和spring-mvc过滤数据库行

徐洋
2023-03-14

为此,我实现了以下控制器:

@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输入框的html" target="_blank">表单,可以用来过滤旅行。

除了lastName之外,我的Travel域对象还有很多我想要筛选的属性。我认为,如果这些属性都是字符串,那么我可以将它们添加为@requestparam并添加一个spring-data-jpa方法来查询这些属性。例如,我会添加一个方法FindByLastNameLikeAndFirstNameLikeAndShipNameLike

我知道,如果我在存储库中实现两个方法,并对控制器使用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出色的回答,在实际实现后,我想为问题的完整性提供一些评论/asnwer:

>

  • 不应实现JPaspecificationExecutor,而应实现JPaspecificationExecutor 以避免类型检查警告。

    请看一下Kostja对这个问题的出色回答,真正的动态JPA CriteriaBuilder,因为如果您想拥有正确的过滤器,就需要实现它。

    我能找到的关于标准API的最好的文档是http://www.ibm.com/developerworks/library/j-typesafejpa/。这是一本相当长的书,但我完全推荐它--读完后,我对鲁特和CriteriaBuilder的大多数问题都得到了回答:)

    重用Travel对象是不可能的,因为它包含各种其他对象(这些对象也包含其他对象),我需要使用,比如,来搜索这些对象--相反,我使用了TravelSearch对象,其中包含了我需要搜索的字段。

    更新10/05/15:根据@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;
    }
    

    希望我能帮上忙!

  • 共有1个答案

    冀越
    2023-03-14

    对于初学者,您应该停止使用@requestparam,并将所有搜索字段放在一个对象中(可能为此重用Travel对象)。然后,您可以使用2个选项来动态构建查询

    1. 使用JPaspecificationExecutor并编写规范
    2. 使用QueryDSLPredicateExecutor并使用QueryDSL编写谓词。

    首先,将JPaspecificationExecutor添加到TravelRepository,这将为您提供FindAll(规范)方法,您可以删除自定义查找器方法。

    public interface TravelRepository extends JpaRepository<Travel, Long>, JpaSpecificationExecutor<Travel> {}
    
    public class TravelSpecification implements Specification<Travel> {
    
        private final Travel criteria;
    
        public TravelSpecification(Travel criteria) {
            this.criteria=criteria;
        }
    
        public Predicate toPredicate(Root<T> root, CriteriaQuery<?> query, CriteriaBuilder builder) {
            // create query/predicate here.
        }
    }
    

    最后,您需要修改控制器以使用新的FindAll方法(我冒昧地稍微清理了一下)。

    @RequestMapping("/search")  
    public String search(@ModelAttribute Travel search, Pageable pageable, Model model) {  
    Specification<Travel> spec = new TravelSpecification(search);
        Page<Travel> travels  = travelRep.findAll(spec, pageable);
        model.addObject("page", new PageWrapper(travels, "/search"));
        return "travels/list";
    }
    

    首先,将QueryDSLPredicateExecutor添加到TravelRepository,这将为您提供一个FindAll(Predicate)方法,您可以删除自定义查找器方法。

    public interface TravelRepository extends JpaRepository<Travel, Long>, QueryDslPredicateExecutor<Travel> {}
    

    接下来,您将实现一个服务方法,该方法将使用travel对象使用QueryDSL构建谓词。

    @Service
    @Transactional
    public class TravelService {
    
        private final TravelRepository travels;
    
        public TravelService(TravelRepository travels) {
            this.travels=travels;
        }
    
        public Iterable<Travel> search(Travel criteria) {
    
            BooleanExpression predicate = QTravel.travel...
            return travels.findAll(predicate);
        }
    }
    
     类似资料:
    • 问题内容: 我有一个使用spring-data-jpa进行数据访问的spring-mvc项目。我有一个域对象,我希望允许最终用户对其应用一些过滤器。 为此,我实现了以下控制器: 效果很好:用户有一个带有输入框的表单,可用于过滤旅行记录。 除了lastName之外,我的域对象还具有许多我想用来过滤的属性。我认为如果这些属性都是字符串,那么我可以将它们添加为s,并添加spring-data-jpa方法

    • 我对Spring和冬眠是新的。我有一个使用Spring、Hibernate和PostgreSQL的项目。我知道我可以使用Spring Data JPA或Hibernate查询数据库,但我不知道每种方法的优点和缺点是什么。我还知道在后台,Spring Data JPA会调用Hibernate。那么我应该使用什么最佳方式来查询数据库以获得最佳性能,Spring数据JPA或Hibernate或取决于具体

    • 我正在尝试使用注解和Spring-安全为我的开源项目添加方法级别的安全性。我现在面临的问题是findAll方法,尤其是用于分页的方法(例如返回页面)。 使用@PostFilter可以在列表上工作(但我个人认为在应用程序而不是数据库中过滤不是一个好主意),但在分页查询上完全失败。 这是有问题的,因为我有一个包含

    • 我有一个带有搜索表单(带有姓名字段)和人员表的网页。当用户插入姓名和姓氏,我必须显示姓名、姓氏的人。当用户只插入姓名时,我必须只按姓名过滤,当用户只输入姓氏时也是如此。 控制器中的我的方法如下所示: 我写了一类: 现在,如果我有名字和姓氏,我的搜索就可以了。问题是在网络表单中,名字或姓氏没有值。 在这种情况下,最佳做法是什么?

    • 我有一个角度应用程序,从带有Spring Data Rest的Spring Boot后端请求数据。对存储库的请求生成带有HATEOAS结构的响应。但是当我从Spring MVC控制器询问时,响应是无关紧要的结构(自然)。 是否有任何示例可以在特定控制器请求上实现相同的结构?我相信我应该实现HATEOAS,但还没有看到一个示例。

    • 29.3.2 Spring Data JPA仓库 Spring Data JPA仓库是可以定义用于访问数据的接口。JPA的查询是根据方法名称自动创建的。例如,CityRepository接口可能会定义一个findAllByState(String state)方法来查找某个州的全部城市。 对于更复杂的查询,您可以使用Spring Data的Query对方法进行注解。 Spring Data仓库通常