本文共 4811 字,大约阅读时间需要 16 分钟。
项目中使用jpa ,第一次见查询起来一脸蒙,这就去查下jpa查询的方式,和概念。
概念
创建使用Java Persistence API的存储库是一个繁琐的过程,需要大量时间并需要大量样板代码。一种推荐的方式是使用元概念
在JPA中,标准查询是以元模型的概念为基础的,元模型是为具体持久化单元的受管实体定义的.这些实体可以是实体类,嵌入类或者映射的父类.提供受管实体元信息的类就是元模型类. 简单的说就是元模型是实体类对应的一个“受管实体 - 举个例子: 实体类 Employee(com.demo.entities包中定义)@Entity@Tablepublic class Employee{ private int id; private String name; private int age; @OneToMany private List addresses; // Other code… }
Employee类的标准元模型类的名字是 Employee_
import javax.annotation.Generated;import javax.persistence.metamodel.SingularAttribute;import javax.persistence.metamodel.ListAttribute;import javax.persistence.metamodel.StaticMetamodel;@StaticMetamodel(Employee.class) public class Employee_ { public static volatile SingularAttributeid; public static volatile SingularAttribute age; public static volatile SingularAttribute name; public static volatile ListAttribute addresses; }
Employee的每一个属性都会使用在JPA2规范中描述的以下规则在相应的元模型类中映射:
- 元模型类的属性全部是static和public的。
元模型类的属性全部是static和public的。Employee的每一个属性都会使用在JPA2规范中描述的以下规则在相应的元模型类中映射:
对于Addess这样的集合类型,会定义静态属性ListAttribute< A, B> b,这里List对象b是定义在类A中类型B的对象。其它集合类型可以是SetAttribute, MapAttribute 或 CollectionAttribute 类型。
看到这应该会有个疑问,这麻烦,为什么要使用这个元模型?有啥好处?
好处肯定是有的,毕竟是标准jpa定义的东西。我这网上查了下,好处很多:
- 查询更加类型安全
好吧,我暂时就查到这个。
为了更好的理解criteria 查询,考虑拥有Employee实例集合的Dept实体,Employee和Dept的元模型类的代码如下:
//All Necessary Imports@StaticMetamodel(Dept.class)public class Dept_ { public static volatile SingularAttributeid; public static volatile ListAttribute employeeCollection; public static volatile SingularAttribute name; } //All Necessary Imports @StaticMetamodel(Employee.class) public class Employee_ { public static volatile SingularAttribute id; public static volatile SingularAttribute age; public static volatile SingularAttribute name; public static volatile SingularAttribute deptId; }
下面的代码片段展示了一个criteria 查询,它用于获取所有年龄大于24岁的员工:
CriteriaBuilder criteriaBuilder = em.getCriteriaBuilder();CriteriaQuerycriteriaQuery = criteriaBuilder.createQuery(Employee.class); Root employee = criteriaQuery.from(Employee.class); Predicate condition = criteriaBuilder.gt(employee.get(Employee_.age), 24); criteriaQuery.where(condition); TypedQuery typedQuery = em.createQuery(criteriaQuery); List result = typedQuery.getResultList();
对应的SQL: SELECT * FROM employee WHERE age > 24
CriteriaBuilder 安全查询创建工厂,,创建CriteriaQuery,创建查询具体具体条件Predicate 等。
CriteriaBuilder是一个工厂对象,安全查询的开始.用于构建JPA安全查询.可以从EntityManager 或 EntityManagerFactory类中获得CriteriaBuilder。 比如:CriteriaBuilder criteriaBuilder = em.getCriteriaBuilder();
CriteriaBuilder criteriaBuilder = em.getCriteriaBuilder();CriteriaQuerycriteriaQuery = criteriaBuilder.createQuery(Employee.class);
Rootemployee = criteriaQuery.from(Employee.class);
Predicate condition = criteriaBuilder.gt(employee.get(Employee_.age), 24); criteriaQuery.where(condition);
过Employee_元模型类age属性,称之为路径表达式。若age属性与String文本比较,编译器会抛出错误,这在JPQL中是不可能的。这就是元模型的作用吗??
ListpredicatesList = new ArrayList ();
predicatesList.add(……Pridicate….)
criteriaQuery.where(predicatesList.toArray(new Predicate[predicatesList.size()]));
OR语句(可怕)predicatesList.add(criteriaBuilder.or(criteriaBuilder.equal(root.get(RepairOrder_.localRepairStatus), LocalRepairStatus.repairing),criteriaBuilder.equal(root.get(RepairOrder_.localRepairStatus), LocalRepairStatus.diagnos)));忽略大小写(全大写)predicatesList.add(criteriaBuilder.like(criteriaBuilder.upper(root.get(RepairShop_.shopName)), StringUtils.upperCase(StringUtils.trim(this.shopName)) + “%”));引用原文:
转载地址:http://wuern.baihongyu.com/