JPA之使用JPQL语句进行增删改查

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]

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

您可能感兴趣的文章:

  • SpringBoot JPA实现增删改查、分页、排序、事务操作等功能示例
(0)

相关推荐

  • SpringBoot JPA实现增删改查、分页、排序、事务操作等功能示例

    今天给大家介绍一下SpringBoot中JPA的一些常用操作,例如:增删改查.分页.排序.事务操作等功能. 下面先来介绍一下JPA中一些常用的查询操作: //And --- 等价于 SQL 中的 and 关键字,比如 findByHeightAndSex(int height,char sex): public List<User> findByHeightAndSex(int height,char sex); // Or --- 等价于 SQL 中的 or 关键字,比如 findByHei

  • JPA之使用JPQL语句进行增删改查

    JPA支持两种表达查询的方法来检索实体和来自数据库的其他持久化数据:查询语句(Java Persistence Query Language,JPQL)和条件API(criteria API).JPQL是独立于数据库的查询语句,其用于操作逻辑上的实体模型而非物理的数据模型.条件API是根据实体模型构建查询条件 1.Java持久化查询语句入门 复制代码 代码如下: List<Person> persons= entityManager.createQuery("select p fro

  • mybatis通过if语句实现增删改查操作

    有时候为了简化我们的代码. 1 举个例子 Student类: @Data public class Student { private Integer id; private Integer age; private Integer sno; } 有时候我们想通过age这个属性获取Student对象 有时候我们也想通过sno这个属性获取Student对象 难道我们在DAO层写两个接口? 比如这样子? Student getStudentByAge(Int age); Student getStu

  • SpringBoot+MySQL+Jpa实现对数据库的增删改查和分页详解

    一. 使用Springboot+Jpa实现对mysql数据库的增删改查和分页功能 JPA是Java Persistence API的简称,中文名Java持久层API,是JDK 5.0注解或XML描述对象-关系表的映射关系,并将运行期的实体对象持久化到数据库中. 使用Springboot和jpa对数据库进行操作时,能够大大减少我们的工作量,在jpa中,已经在底层封装好了增删查的功能和sql语句,可以使我们进行快速开发 二.项目过程和配置文件 1.applaction.properties文件配置

  • 基于sqlalchemy对mysql实现增删改查操作

    需求场景: 老大让我利用爬虫爬取的数据写到或更新到mysql数据库中,百度了两种方法 1 是使用pymysql连接mysql,通过操作原生的sql语句进行增删改查数据: 2 是使用sqlalchemy连接mysql,通过ORM模型建表并操作数据库,不需要写原生的sql语句,相对简单些: 以下就是本次使用sqlalchemy的经验之谈. 实现流程:连接数据库>通过模型类创建表>建立会话>执行创建表语句>通过会话进行增删改查 from sqlalchemy import exists,

  • MySQL中增删改查操作与常见陷阱详解

    目录 本文导读 一.MySQL的增删改查 1.insert语句 2.delete语句 3.update语句原理 4.select 二.15种MySQL数据操作语句 1.REPLACE语句 2.CALL语句 3.TABLE语句 4.WITH语句 三.MySQL查询陷阱 总结 本文导读 本文作为MySQL系列第二篇文章,详细讲解了MySQL的增删改查的语句.语义和一些我们经常在开发工作中暴露的问题,MySQL的增删改查又叫数据操作语句,本文有讲些了一些常用的数据操作语句,select语句后续将作为一

  • 使用jpa原生sql@Query操作增删改查

    jpa原生sql@Query操作增删改查 1.jpa 原生update的sql语句: 1.命名参数(推荐使用此方式):可以定义好参数名,赋值时使用@Param("参数 名"), 而不用管顺序. 为@Query注解传递参数的方式1:命名参数 /*@Query("delete from product where p.id=:id") List<Person> testQueryAnnotationParams2(@Param("status&qu

  • 利用python模拟sql语句对员工表格进行增删改查

    本文主要给大家介绍了关于python模拟sql语句对员工表格进行增删改查的相关内容,分享出来供大家参考学习,下面来一起看看详细的介绍: 具体需求: 员工信息表程序,实现增删改查操作: 可进行模糊查询,语法支持下面3种: select name,age from staff_data where age > 22                  多个查询参数name,age 用','分割 select * from staff_data where dept = 人事 select * from

  • mysql增删改查基础语句

    语法 这里是INSERT INTO命令将数据插入到MySQL表的通用SQL语法: INSERT INTO table_name ( field1, field2,...fieldN ) VALUES ( value1, value2,...valueN ); 要插入字符串类型数据,则需要双或单引号保留到所有的值,例如: "value". 1.从命令提示符插入数据 这将使用SQL 的INSERT INTO命令将数据插入到MySQL表:tutorials_tbl 示例 root@host#

  • Mysql的增删改查语句简单实现

    Mysql的增删改查语句简单实现 增加记录: insert into tablename(...) values(...) //如果增加的记录包括所有的列,则不需要写数据列表 insert into tablename values(...) 删除记录: delete from tablename where condition ; 修改记录: update tablename set xx=xx , xx=xx... where condition ; alter table tablenam

随机推荐