JPA之使用JPQL进行CRUD操做

JPA支持两种表达查询的方法来检索实体和来自数据库的其余持久化数据:查询语句(Java Persistence Query Language,JPQL)和条件API(criteria API)。JPQL是独立于数据库的查询语句,其用于操做逻辑上的实体模型而非物理的数据模型。条件API是根据实体模型构建查询条件java

1.Java持久化查询语句入门 List persons= entityManager.createQuery("select p from Person p").getResultList();数据库

1.这个查询语句相似于SQL。但它与真正的SQL的区别是,它不是从一个表中进行选择查询,而是指定来自应用程序域模型的实体。     2.查询select子句也只是列出了查询实体的别名,若是只查询某一列的,可使用点(.)操做符进行来导航实体属性。以下所示: List persons= entityManager.createQuery("select p.firstName from Person p").getResultList();app

1.1.筛选条件     像SQL同样,JPQL也支持where子句,用于对搜索的条件过滤。包括大多数的操做符,如:in,between、like以及函数表达式substring、length等等 List persons = entityManager.createQuery("select p from Person p where p.age>23").getResultList();函数

1.2.投影结果     对于查询的数据量比较大的话,可使用投影的方式,只查询出有用的列。 //投影 List persons = entityManager.createQuery("select p.firstName,p.age from Person p").getResultList();工具

1.3.聚合查询     JPQL的聚合查询语法相似于SQL。例如count List count=entityManager.createQuery("select count(p) from Person p").getResultList();hibernate

1.4.查询参数     JPQL支持两种类型的参数绑定语法。对象

1.位置参数表示法接口

其中参数是在查询字符串中指示,该字符串是在一个问号(?)以后紧随参数的编号。当执行查询的时候,开发人员指定应该替换的参数编ip

Query query=entityManager.createQuery("select p from Person p where p.age=?1 and p.firstName=?2"); query.setParameter(1,21); query.setParameter(2,"Jack");开发

2.命名参数表示法

经过在一个冒号(:)以后紧随参数名称,在查询字符串对它进行指示,当执行查询的时候,开发人员指定应该替换的参数名称

Query query=entityManager.createQuery("select p from Person p where p.age=:age and p.firstName=:name"); query.setParameter("age",21); query.setParameter("name","Jack");

2.定义查询     JPA提供Query和TypedQuery(JPA 2.0引入)接口来配置和执行查询。Query的返回的Object类型,而TypedQuery返回的是指定的Class类型。 //未指定类型,返回Object类型 Query q = entityManager.createQuery("select p from Person p"); //指定返回类型为Person类型 TypedQuery q1 = entityManager.createQuery("select p from Person p", Person.class);

2.1.动态查询定义     JPA查询引擎,能够将JPQL字符串解析成语法树,获取表达式中的实体对象-关系映射的元数据,而后生成等价的SQL。故有两种方式进行动态查询。

1.拼接字符串方式

Tip:会引发SQL注入问题

/**

  • 动态拼接字符串构建查询条件
  • @param name
  • @param age
  • @return */ public static String queryPersonJPQL(String name, int age) { String queryQL = "select p from Person p where p.firstName= '" + name + "' and p.age=" + age; return queryQL; }

//调用 Query query = entityManager.createQuery(queryPersonJPQL("jack", 21));

2.动态参数化构建查询条件(推荐使用)

/**

  • 动态参数化构建查询条件
  • @return */ public static String queryPersonJPQLByParams() { String queryQL = "select p from Person p where p.firstName=:name and p.age=:age"; return queryQL; }

Query query = entityManager.createQuery(queryPersonJPQLByParams()); query.setParameter("name", "Jack"); query.setParameter("age", 21);

2.2.命名查询定义     命名查询是一个强大的工具。使用@NamedQuery注解定义一个命名查询,能够把它放在任何实体的类定义之上。该注解定义了查询的名称,及其查询的文本。

Tip:命名查询通畅放置在对应查询结果的实体类上

@Entity @NamedQuery(name = "findByAge", query = "select p from Person p where p.age=:age") public class Person { //省略 }

Tip:NamedQuery里面定义的名称在整个持久化单元中须要惟一,否则运行会出错。

eg: Exception in thread "main" org.hibernate.DuplicateMappingException: Duplicate query mapping findByAge at org.hibernate.boot.internal.InFlightMetadataCollectorImpl.checkQueryName

调用

List people = entityManager.createNamedQuery("findByAge", Person.class).setParameter("age", 21).getResultList();

若是一个类定义两个或者以上个的命名查询,那么必须把它放置在@NamedQueries()

2.3.绑定参数     经过前面的例子,咱们能够看到绑定参数有两种方式:1.位置参数化绑定。2.命名参数化绑定。都是经过Query接口的setParameter方法进行绑定。

1.位置参数化

TypedQuery setParameter(int position, Object value);

2.命名参数化

TypedQuery setParameter(String name, Object value);

第一种位置参数化绑定,若是位置发生变化都须要改变绑定的代码。推荐使用第二种。

2.4.执行查询     Query接口与TypedQuery接口提供了三种不一样的方式执行查询。

1.executeUpdate

用来执行批量更新或者删除

2.getSingleResult

获取单个结果集。若是没有获取到数据,则会抛出NoResultException异常。若是获取多条数据的话,则会抛出NonUniqueResultException异常

3.getResultList

获取对应的结果集合,指定顺序的集合,须要使用List做为返回值类型。若是没有获取到数据的话,则返回一个空集合,不会抛出异常

2.5.分页     经过setFirstResult() 和setMaxResults() 方法能够完成分页的查询

查询页码为0,每页展现2条数据

List people = entityManager.createQuery("select p from Person p ", Person.class).setFirstResult(0).setMaxResults(2).getResultList();

Tip:不能用于经过集合关系链接的查询,由于这些查询可能返回重复的值。

2.6.查询超时     若是一个应用程序须要设置查询响应时间的限制,那么能够在查询中设置javax.persistence.query.timeout属性(jpa 2.0引入)或者将它做为持久化属性的一部分。此属性定义了查询在终止前容许容许运行的==毫秒数==。若是查询超时的时候,会抛出QueryTimeoutException。

TypedQuery query = entityManager.createQuery("select p from Person p", Person.class);

//单位为毫秒 javax.persistence.query.timeout query.setHint("javax.persistence.query.timeout", 5000); List people = query.getResultList();

2.7.批量更新和删除     批量更新实体是经过update语句完成。批量删除实体是经过delete语句完成。二者皆指定的是实体及其类的属性。

entityManager.getTransaction().begin(); Query query = entityManager.createQuery("update Person p set p.firstName=:name where p.id=:id"); query.setParameter("name", "xiaobai"); query.setParameter("id", 2); query.executeUpdate();

Query query1 = entityManager.createQuery("delete Person p where p.id=:id"); query1.setParameter("id", 9); query1.executeUpdate(); entityManager.getTransaction().commit();

3.使用JPQL查询的建议     在应用系统中,一般使用查询的次数要比增长、修改、删除要多。故合理的使用查询显的尤其重要。

1.建议采用命名查询(NamedQuery)

持久化提供的程序一般会采用预编译的方式将命名查询做为程序初始化阶段的一部分。这样就避免了连续解析JPQL和生成SQL的系统开销。

2.大数量优先使用投影方式检索少许的列

jpa查询一般返回的是整个实体的全部列,可是对于庞大的数据量而言,并非全部的实体列都须要用到。那么咱们可使用投影的方式来处理。

List<List<Object[]>> persons = entityManager.createQuery("select new List(firstName,age) from Person p").getResultList(); for (Object o : persons) { System.out.println(o); }

//输出结果

[Jack, 21]

[Jack, 21]

[Jack, 21]

[lily, 19]

[tom, 23]

[tom, 23]

相关文章
相关标签/搜索