JPA支持两种表达查询的方法来检索实体和来自数据库的其他持久化数据:查询语句(Java Persistence Query Language,JPQL)和条件API(criteria API)。JPQL是独立于数据库的查询语句,其用于操作逻辑上的实体模型而非物理的数据模型。条件API是根据实体模型构建查询条件
1.Java持久化查询语句入门
List<Person> persons= entityManager.createQuery("select p from Person p").getResultList();
1.这个查询语句类似于SQL。但它与真正的SQL的区别是,它不是从一个表中进行选择查询,而是指定来自应用程序域模型的实体。
2.查询select子句也只是列出了查询实体的别名,如果只查询某一列的,可以使用点(.)操作符进行来导航实体属性。如下所示:
List<String> persons= entityManager.createQuery("select p.firstName from Person p").getResultList();
1.1.筛选条件
像SQL一样,JPQL也支持where子句,用于对搜索的条件过滤。包括大多数的操作符,如:in,between、like以及函数表达式substring、length等等
List<Person> persons = entityManager.createQuery("select p from Person p where p.age>23").getResultList();
1.2.投影结果
对于查询的数据量比较大的话,可以使用投影的方式,只查询出有用的列。
//投影List<Object> persons = entityManager.createQuery("select p.firstName,p.age from Person p").getResultList();
1.3.聚合查询
JPQL的聚合查询语法类似于SQL。例如count
List<Integer> count=entityManager.createQuery("select count(p) from Person p").getResultList();
1.4.查询参数
JPQL支持两种类型的参数绑定语法。
1.位置参数表示法
其中参数是在查询字符串中指示,该字符串是在一个问号(?)之后紧随参数的编号。当执行查询的时候,开发人员指定应该替换的参数编
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<Person> 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<Person> people = entityManager.createNamedQuery("findByAge", Person.class).setParameter("age", 21).getResultList();
如果一个类定义两个或者以上个的命名查询,那么必须把它放置在@NamedQueries()
2.3.绑定参数
通过前面的例子,我们可以看到绑定参数有两种方式:1.位置参数化绑定。2.命名参数化绑定。都是通过Query接口的setParameter方法进行绑定。
1.位置参数化
TypedQuery<X> setParameter(int position, Object value);
2.命名参数化
TypedQuery<X> 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<Person> 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<Person> query = entityManager.createQuery("select p from Person p", Person.class); //单位为毫秒 javax.persistence.query.timeout query.setHint("javax.persistence.query.timeout", 5000); List<Person> 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]
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持小牛知识库。
问题内容: 这是SQL中的JOIN问题更新语句的扩展,但是我尝试使用Spring Data JPQL。 我正在尝试将更新与JPQL中的JOIN一起使用,如下所示 但是,我得到如下错误 org.hibernate.hql.internal.ast.QuerySyntaxException:期望“设置”,找到“ JOIN” JPQL中无法进行UPDATE和JOIN吗?有什么选择。谢谢 问题答案: 该J
我有一个连接两个表的非常简单的连接表,为了简化问题,我们可以说表1是a,表2是B,有一个连接表AB。 其中A_id_fk和B_id_fk分别是外键。我试图创建一个查询来检索a中与B有关系的所有行,所以我的函数接收B_id作为参数,我想搜索AB以获得B_id_fk==B_id的所有行,然后使用搜索a搜索A_id==上一次搜索返回的A_id_fk的所有行。 我进行了测试,可以在简单的SQL中使用嵌套的
问题内容: 这是SQL中的JOIN问题更新语句的扩展,但是我试图使用Spring Data JPQL。 我正在尝试将更新与JPQL中的JOIN一起使用,如下所示 但是,我得到如下错误 org.hibernate.hql.internal.ast.QuerySyntaxException:期望“设置”,找到“ JOIN” JPQL中无法进行UPDATE和JOIN吗?有什么选择。谢谢 问题答案: 该J
这是在SQL中使用JOIN的question Update语句的扩展,但我正在尝试使用Spring Data JPQL。 我试图在JPQL中使用Update和JOIN,如下所示 在JPQL中是否不能更新和连接?另一种选择是什么。谢谢
本文向大家介绍mybatis使用xml进行增删改查代码解析,包括了mybatis使用xml进行增删改查代码解析的使用技巧和注意事项,需要的朋友参考一下 MyBatis是支持普通sql查询、存储过程和高级映射的持久层框架。 MyBatis消除了几乎所有的JDBC代码和参数的手工设置以及对结果集的检索封装。 MyBatis可以使用 简单的XML或注解用于配置和原始映射,将接口和Java的POJO(Pl
问题内容: 我想将此SQL语句转换为JPQL等效项。 这样可以正确地从表中检索信息。 在我的实体中 到目前为止,这是我所拥有的,但是它不起作用。 我究竟做错了什么?谢谢。 问题答案: 试试这个查询(替换用):