springdataJPA 使用Predicate Join 做四表复杂条件查询(分页)

近期在做项目时,遇到一种表关系,很复杂的分页,在这中间尝试过很多方法都是不行,最终通过尝试找到如下解决方案,希望可以帮到有需要的人!

首先说一下表关系:

设备数据表:

设备表:

用户表:

要实现效果为:通过小区名称 或者楼栋名称区分 设备数据.解决方案如下:

  Specification<RoomTempHum> specification = new Specification<RoomTempHum>() {
            @Override
            public Predicate toPredicate(Root<RoomTempHum> root, CriteriaQuery<?> criteriaQuery, CriteriaBuilder cb) throws Exception {
                List<Predicate> list = new ArrayList<>();
                //链接设备
                Join<RoomTempHum, RoomEquipment> roomEquipmentJoin = root.join("roomEquipment", JoinType.LEFT);
                //设备ID
                if (!StringUtils.isEmpty(roomTempHum.getEquipmentId())) {
                    Predicate equipmentId = cb.equal(root.get("equipmentId"), roomTempHum.getEquipmentId());
                    list.add(equipmentId);
                }
                //设备不为null
                if (roomTempHum.getRoomEquipment() != null) {
                    //住户不为null
                    if (roomTempHum.getRoomEquipment().getHouseHold() != null) {
                        if (!StringUtils.isEmpty(roomTempHum.getRoomEquipment().getHouseHoldId())) {
                            Predicate predicate = cb.equal(roomEquipmentJoin.get("houseHold").get("id"), roomTempHum.getRoomEquipment().getHouseHold().getId());
                            list.add(predicate);
                        }
                        //小区不为null
                        if (!StringUtils.isEmpty(roomTempHum.getRoomEquipment().getHouseHold().getCommunity().getId() )) {
                            Predicate id = cb.equal(roomEquipmentJoin.get("houseHold").get("community").get("id"), roomTempHum.getRoomEquipment().getHouseHold().getCommunity().getId()
                            );
                            list.add(id);
                        }
                        //楼栋不为null
                        if (!StringUtils.isEmpty(roomTempHum.getRoomEquipment().getHouseHold().getBuilding().getId())) {
                            Predicate id = cb.equal(roomEquipmentJoin.get("houseHold").get("building").get("id"), roomTempHum.getRoomEquipment().getHouseHold().getBuilding().getId()
                            );
                            list.add(id);
                        }
                    }
                }
                //业务日期筛选
                if (roomTempHum.getStartToEnd() != null && roomTempHum.getStartToEnd().get(0) != null && roomTempHum.getStartToEnd().get(1) != null) {
                    Predicate re_time = cb.between(root.get("reTime").as(String.class), convertDate(roomTempHum.getStartToEnd().get(0)), convertDate(roomTempHum.getStartToEnd().get(1)));
                    list.add(re_time);
                }

                if (list.size() == 0) {
                    return null;
                } else {
                    Predicate[] p = new Predicate[list.size()];
                    return cb.and(list.toArray(p));
                }
            }
        };

上述代码中的核心解决方案是:

第一步:join 链接设备信息表

第二步:条件:通过join.get("houseHold") 获取到用户,再通过join.get("houseHold").get("community")获取到小区 楼栋也是一样的.

第三步:取值:通过 roomTempHum.getRoomEquipment().getHouseHold().getCommunity().getId() 获取到前端传入的值.(注意:roomTempHum为前端传入的值对象!!!)

通过以上三步,就可以拼接完自己想要实现的条件了,以上方法同样适用于N表查询分页!!!

 

最后,希望对有帮助的人,给点个关注!随时交流

 

 

 

 

 

 

使用Spring Data JPA进行多条件组合条件查询时,可以通过使用Specification来实现。Specification是一个接口,我们可以自定义一个实现该接口的类,然后在查询方法中传入该Specification对象来指定查询条件。 首先,我们需要在仓库接口中定义一个方法,该方法接收一个Specification参数,并返回查询结果。如下所示: ```java @Repository public interface UserRepository extends JpaRepository<User, Long> { List<User> findAll(Specification<User> spec); } ``` 接下来,我们在Specification实现类中重写toPredicate方法,在该方法中使用CriteriaBuilder构建查询条件,并返回一个Predicate对象查询条件。例如,我们可以按用户名和年龄进行查询,如下所示: ```java public class UserSpecification implements Specification<User> { private String username; private int age; public UserSpecification(String username, int age) { this.username = username; this.age = age; } @Override public Predicate toPredicate(Root<User> root, CriteriaQuery<?> query, CriteriaBuilder criteriaBuilder) { List<Predicate> predicates = new ArrayList<>(); if (username != null) { predicates.add(criteriaBuilder.equal(root.get("username"), username)); } if (age != 0) { predicates.add(criteriaBuilder.equal(root.get("age"), age)); } return criteriaBuilder.and(predicates.toArray(new Predicate[0])); } } ``` 最后,我们可以在服务类中调用仓库方法并传入自定义的Specification对象来进行查询。例如: ```java @Service public class UserService { @Autowired private UserRepository userRepository; public List<User> getUsers(String username, int age) { UserSpecification spec = new UserSpecification(username, age); return userRepository.findAll(spec); } } ``` 这样,我们就可以根据传入的条件来进行组合条件查询了。当传入的条件为空时,不会加入到查询中。当传入的条件有值时,则会根据该条件进行查询。这样,就实现了多条件组合条件查询
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值