发表于: 2017-09-30 22:10:00

0 711


  • 今天完成的事情:学习动态查询
  • 一 概念

  • 1.什么是动态查询?

  1. 从多个查询条件中随机选择若干个组合成一个DQL语句进行查询,这一过程叫做动态查询。


2.动态查询的难点

可供选择的查询条件多,组合情况多,难以一一列举。


3.最终查询语句的构成

一旦用户向查询条件中输入数据,该查询条件就成为最终条件的一部分。


二 基本原理

1.SQL基本框架

无论查询条件如何,查询字段与数据库是固定不变的,这些固定不变的内容构成SQL语句的基本框架,如

select column... from table。


2.StringBuilder形成DQL

       获取表单输入,如果请求参数非空,根据该请求参数生成查询条件,如“name=?”,“age>?”,将查询条件追加到基本框架中。利用StringBuilder来追加查询条件,这时出现一个问题,怎么判断生成的查询条件中是否需要添加“and”?

       如果该查询条件是第一个查询条件,不需要添加"and",否则需要添加“and”。问题变得复杂起来,每一次生成查询条件时都需要判断前面是否存在查询条件。

       我们可以考虑在SQL基本框架中添加一个查询条件,该查询条件的存在不影响查询结果,只充当占位角色,避免动态添加查询条件时判断是否需要添加“and”。根据这些要求,这一查询条件必须恒为真,这里我们取“1=1”,SQL基本框架就变成了select column...from table where 1=1每一个动态查询条件前段都添加“and”。


3.List集合为占位符赋值

       有了DQL语句,接着需要考虑怎么为占位符赋值。可以在生成查询条件的同时,将占位符对应的参数收集起来,存入一个有序集合中,这里选择List集合,这样占位符就与List集合中的元素形成了顺序上的对应关系,第n个占位符对应第n个元素,遍历集合就可以为占位符赋值了。

       为占位符赋值时,不仅仅需要将数据传递给占位符,还需要选择与字段一致的数据类型,List集合仅仅存储数据已经不能够满足要求了,还需要添加字段信息,以区分不同的字段,选择不同的数据类型。这里集合中的元素采用“column+data”的形式。


三、代码实战

实体类什么的省略,直接放测试类代码

private Specification<DevHREmpConstrast> where(

  final String corg,final String name,final String type,final String date,final String checker){

 return new Specification<DevHREmpConstrast>() {

  @Override

  public Predicate toPredicate(Root<DevHREmpConstrast> root, CriteriaQuery<?> query, CriteriaBuilder cb) {

   List<Predicate> predicates = new ArrayList<Predicate>();

   //机构

   if(corg!=null&&!corg.equals("")){

    List<String> orgIds = organizationDao.findByName("%"+corg+"%");

    if(orgIds.size()>0&&orgIds.size()<1000)

     predicates.add(root.<String>get("confirmOrgNo").in(orgIds));//confirmOrgNo

   }

   //名字

   if(name!=null&&!name.equals("")){

    List<String> userIds = userDao.findByName(name);

    if(userIds.size()>0&&userIds.size()<1000)//如果太多就不管了这个条件了

     predicates.add(root.<String>get("hrUserName").in(userIds));

   }

   //类型

   if(type!=null&&!type.equals(""))

    predicates.add(cb.equal(root.<String>get("hrUpdateType"),type));

   //日期

   if(date!=null&&!date.equals("")){

    //处理时间

    SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd");

    Date startDate;

    Date endDate;

    try {

     startDate = format.parse(date);

    } catch (ParseException e) {

     startDate = new Date(946656000000L);//2000 01 01

    }

    endDate = startDate;

    Calendar calendar = Calendar.getInstance() ;

    calendar.setTime(endDate);

    calendar.add(Calendar.DATE, 1);

    endDate = calendar.getTime();

    calendar = null;

    predicates.add(cb.between(root.<Date>get("insDate"),startDate,endDate));

   }

   //审核人

   if(checker!=null&&!checker.equals("")){

    List<String> userIds = userDao.findByName(checker);

    if(userIds.size()>0&&userIds.size()<1000)//如果太多就不管了这个条件了

     predicates.add(root.<String>get("confirmUserId").in(userIds));

   }

   return query.where(predicates.toArray(new Predicate[predicates.size()])).getRestriction();

  }

 };

}

明天计划的事情:看老大讲解视屏

遇到的问题:动态查询不熟

收获:学习动态查询


返回列表 返回列表
评论

    分享到