JPA之使⽤JPQL语句进⾏增删改查
JPA⽀持两种表达查询的⽅法来检索实体和来⾃数据库的其他持久化数据:查询语句(Java Persistence Query Language,JPQL)和条件API(criteria API)。JPQL是独⽴于数据库的查询语句,其⽤于操作逻辑上的实体模型⽽⾮物理的数据模型。条件API是根据实体模型构建查询条件
1.Java持久化查询语句⼊门
复制代码代码如下:
List<Person> persons= ateQuery("select p from Person p").getResultList();
1.这个查询语句类似于SQL。但它与真正的SQL的区别是,它不是从⼀个表中进⾏选择查询,⽽是指定来⾃应⽤程序域模型的实体。
2.查询select⼦句也只是列出了查询实体的别名,如果只查询某⼀列的,可以使⽤点(.)操作符进⾏来导航实体属性。如下所⽰:复制代码代码如下:
List<String> persons= ateQuery("select p.firstName from Person p").getResultList();
1.1.筛选条件
像SQL⼀样,JPQL也⽀持where⼦句,⽤于对搜索的条件过滤。包括⼤多数的操作符,如:in,between、like以及函数表达式substring、length等等
复制代码代码如下:
List<Person> persons = ateQuery("select p from Person p where p.age>23").getResultList();
1.2.投影结果
对于查询的数据量⽐较⼤的话,可以使⽤投影的⽅式,只查询出有⽤的列。
复制代码代码如下:
//投影List<Object> persons = ateQuery("select p.firstName,p.age from Person p").getResultList();
1.3.聚合查询
JPQL的聚合查询语法类似于SQL。例如count
复制代码代码如下:
List<Integer> ateQuery("select count(p) from Person p").getResultList();
1.4.查询参数
JPQL⽀持两种类型的参数绑定语法。
1.位置参数表⽰法
其中参数是在查询字符串中指⽰,该字符串是在⼀个问号(?)之后紧随参数的编号。当执⾏查询的时候,开发⼈员指定应该替换的参数编
Query ateQuery("select p from Person p where p.age=?1 and p.firstName=?2");
query.setParameter(1,21);
query.setParameter(2,"Jack");
2.命名参数表⽰法
通过在⼀个冒号(:)之后紧随参数名称,在查询字符串对它进⾏指⽰,当执⾏查询的时候,开发⼈员指定应该替换的参数名称
Query ateQuery("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 = ateQuery("select p from Person p");
//指定返回类型为Person类型
TypedQuery<Person> q1 = ateQuery("select p from Person p", Person.class);
2.1.动态查询定义
JPA查询引擎,可以将JPQL字符串解析成语法树,获取表达式中的实体对象-关系映射的元数据,然后⽣成等价的SQL。故有两种⽅式进⾏动态查询。
1.拼接字符串⽅式
Tip:会引起SQL注⼊问题
/**
* 动态拼接字符串构建查询条件
*
* @param name
* @param age
批量更新sql语句* @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 = ateQuery(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 = ateQuery(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 = ateNamedQuery("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接⼝提供了三种不同的⽅式执⾏查询。
⽤来执⾏批量更新或者删除
获取单个结果集。如果没有获取到数据,则会抛出NoResultException异常。如果获取多条数据的话,则会抛出NonUniqueResultException异常
获取对应的结果集合,指定顺序的集合,需要使⽤List作为返回值类型。如果没有获取到数据的话,则返回⼀个空集合,不会抛出异常
2.5.分页
通过setFirstResult() 和setMaxResults() ⽅法可以完成分页的查询
查询页码为0,每页展⽰2条数据
复制代码代码如下:
List<Person> people = ateQuery("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 = ateQuery("select p from Person p", Person.class);
//单位为毫秒 javax.persistence.query.timeout
query.setHint("javax.persistence.query.timeout", 5000);
List<Person> people = ResultList();
2.7.批量更新和删除
批量更新实体是通过update语句完成。批量删除实体是通过delete语句完成。两者皆指定的是实体及其类的属性。Transaction().begin();
Query query = ateQuery("update Person p set p.firstName=:name where p.id=:id");
query.setParameter("name", "xiaobai");
query.setParameter("id", 2);
Query query1 = ateQuery("delete Person p where p.id=:id");
query1.setParameter("id", 9);
3.使⽤JPQL查询的建议
在应⽤系统中,通常使⽤查询的次数要⽐增加、修改、删除要多。故合理的使⽤查询显的尤为重要。
1.建议采⽤命名查询(NamedQuery)
持久化提供的程序通常会采⽤预编译的⽅式将命名查询作为程序初始化阶段的⼀部分。这样就避免了连续解析JPQL和⽣成SQL的系统开销。
2.⼤数量优先使⽤投影⽅式检索少量的列
jpa查询通常返回的是整个实体的所有列,但是对于庞⼤的数据量⽽⾔,并不是所有的实体列都需要⽤到。那么我们可以使⽤投影的⽅式来处理。
List<List<Object[]>> persons = ateQuery("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]
以上就是本⽂的全部内容,希望对⼤家的学习有所帮助,也希望⼤家多多⽀持。

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。