JPA初级入门学习笔记

学习尚硅谷JPA教程
JPA是一个用于对象持久化的 API , 是一个ORM规范
在这里插入图片描述

JPA和Hibernate关系

  • JPA 是 hibernate 的一个抽象(就像JDBC和JDBC驱动的关系):
  • JPA 是规范:JPA 本质上就是一种 ORM 规范,不是ORM 框架 —— 因为 JPA 并未提供 ORM 实现,它只是制订了一些规范,提供了一些编程的 API 接口,但具体实现则由 ORM 厂商提供实现
  • Hibernate 是实现:Hibernate 除了作为 ORM 框架之外,它也是一种 JPA 实现
    从功能上来说, JPA 是 Hibernate 功能的一个子集

JPA优势

  • 标准化: 提供相同的 API,这保证了基于JPA 开发的企业应用能够经过少量的修改就能够在不同的 JPA 框架下运行。
  • 简单易用,集成方便: JPA 的主要目标之一就是提供更加简单的编程模型,在 JPA 框架下创建实体和创建 Java 类一样简单,只需要使用 javax.persistence.Entity 进行注释;JPA 的框架和接口也都非常简单,
  • 可媲美JDBC的查询能力: JPA的查询语言是面向对象的,JPA定义了独特的JPQL,而且能够支持批量更新和修改、JOIN、GROUP BY、HAVING 等通常只有 SQL 才能够提供的高级查询特性,甚至还能够支持子查询。
    支持面向对象的高级特性: JPA 中能够支持面向对象的高级特性,如类之间的继承、多态和类之间的复杂关系,最大限度的使用面向对象的模型

换句话说,为啥只用JPA而不用Hibernate呢?这是因为hibernate对jpa的支持,不是另提供了一套专用于jpa的注解。一些重要的注解如Column, OneToMany等,hibernate没有提供,这说明jpa的注解已经是hibernate 的核心,hibernate只提供了一些补充,而不是两套注解。
hibernate不只实现了JPA的整套功能,还在此基础上增加了一些额外的功能,若JPA提供的功能够我们用了,我们就可以直接用JPA,而且以后切换到另外实现类也很方便,若用Hibernate的话,需要学习一写它的用法,增加了学习成本,而且不利于切换到JPA其他实现版本上去

大概流程

1. 导入相关依赖

尚硅谷JPA教程中使用的是eclipse , 非Maven项目 , 所以是手动导入

  • hibernate-release-4.2.4.Final\lib\required*.jar
  • hibernate-release-4.2.4.Final\lib\jpa*.jar
  • 数据库驱动的 jar 包
2. 配置persistence.xml

JPA主要是基于注解开发 , 有一个核心配置类persistence.xml
JPA 规范要求在类路径的 META-INF 目录下放置
如果在IDEA中,resources文件夹下,也就是资源文件夹下,创建META-INF文件夹,在该文件夹下创建persistence.xml文件,添加以下配置。

<?xml version="1.0" encoding="UTF-8"?>
<persistence version="2.0"
			 xmlns="http://java.sun.com/xml/ns/persistence" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
			 xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com/xml/ns/persistence/persistence_2_0.xsd">
	<persistence-unit name="jpa-1" transaction-type="RESOURCE_LOCAL">

		<!--		配置使用什么 ORM 产品来作为 JPA 的实现 -->
		<!--		1. 实际上配置的是  javax.persistence.spi.PersistenceProvider 接口的实现类-->
		<!--		2. 若 JPA 项目中只有一个 JPA 的实现产品, 则也可以不配置该节点. -->
		<provider>org.hibernate.ejb.HibernatePersistence</provider>

		<!-- 添加持久化类  , 将需要持久化的实体类配置在这里 , 只有将实体类加在这里 , JPA才能对实体类进行操作-->
		<class>com.atguigu.jpa.helloworld.Customer</class>
		<class>com.atguigu.jpa.helloworld.Order</class>

		<class>com.atguigu.jpa.helloworld.Department</class>
		<class>com.atguigu.jpa.helloworld.Manager</class>

		<class>com.atguigu.jpa.helloworld.Item</class>
		<class>com.atguigu.jpa.helloworld.Category</class>

		<!-- 
		配置二级缓存的策略 
		ALL:所有的实体类都被缓存
		NONE:所有的实体类都不被缓存. 
		ENABLE_SELECTIVE:标识 @Cacheable(true) 注解的实体类将被缓存
		DISABLE_SELECTIVE:缓存除标识 @Cacheable(false) 以外的所有实体类
		UNSPECIFIED:默认值,JPA 产品默认值将被使用
		-->
		<shared-cache-mode>ENABLE_SELECTIVE</shared-cache-mode>

		<properties>
			<!-- 连接数据库的基本信息 -->
			<property name="javax.persistence.jdbc.driver" value="com.mysql.jdbc.Driver"/>
			<property name="javax.persistence.jdbc.url" value="jdbc:mysql:///jpa"/>
			<property name="javax.persistence.jdbc.user" value="root"/>
			<property name="javax.persistence.jdbc.password" value="1230"/>

			<!-- 配置 JPA 实现产品的基本属性. 配置 hibernate 的基本属性 -->
			<!-- 格式化Hibernate的SQL输出语句 , 更利于阅读 -->
			<property name="hibernate.format_sql" value="true"/>
			<!-- 在控制台显示hibernate生成的 sql 语句  -->
			<property name="hibernate.show_sql" value="true"/>
			<!-- 加载hibernate自动更新数据库结构 -->
			<property name="hibernate.hbm2ddl.auto" value="update"/>

			<!-- 二级缓存相关 -->
			<property name="hibernate.cache.use_second_level_cache" value="true"/>
			<property name="hibernate.cache.region.factory_class" value="org.hibernate.cache.ehcache.EhCacheRegionFactory"/>
			<property name="hibernate.cache.use_query_cache" value="true"/>
		</properties>
	</persistence-unit>
</persistence>
3. 实体类基于注解写法
//数据库中要创建的对应的表名
@Table(name="JPA_CUTOMERS")
//告诉JPA这是一个需要持久化的类
@Entity
public class Customer {
	private Integer id;
	private String lastName;

	//@Column: 指定数据库中对应的列名 , 如果数据库列名 和 实体字段名一样就可以省略不写
	@Column(name= "ID")
	//@GeneratedValue 用于标注主键的生成策略,通过strategy 属性指定。  GenerationType.AUTO: 系统自动创建
	@GeneratedValue(strategy=GenerationType.AUTO)
	//@Id 标注用于声明一个实体类的属性映射为数据库的主键列
	@Id
	public Integer getId() {
		return id;
	}

	public void setId(Integer id) {
		this.id = id;
	}

	@Column(name="LAST_NAME",length=50,nullable=false)
	public String getLastName() {
		return lastName;
	}

	public void setLastName(String lastName) {
		this.lastName = lastName;
	}

}
4. 创建main方法

参考文章:JPA - Persistence与EntityManagerFactory

或者创建测试类 将注解跑起来

public class Main {
	public static void main(String[] args) {
		//1. 创建 EntitymanagerFactory 接口对象
		//persistence 配置文件 <persistence-unit 标签中的 name 属性
		String persistenceUnitName = "jpa-1";
		
		Map<String, Object> properites = new HashMap<String, Object>();
		properites.put("hibernate.show_sql", true);

		//Persistence 类是用于获取 EntityManagerFactory 实例。该类包含一个名为 createEntityManagerFactory 的 静态方法 。
		EntityManagerFactory entityManagerFactory = 
				//Persistence.createEntityManagerFactory(persistenceUnitName);
				Persistence.createEntityManagerFactory(persistenceUnitName, properites);
				
		//2. 创建 EntityManager实例. 类似于 Hibernate 的 SessionFactory
		//EntityManagerFactory 接口主要用来创建 EntityManager 实例。
		EntityManager entityManager = entityManagerFactory.createEntityManager();
		
		//3. 开启事务
		EntityTransaction transaction = entityManager.getTransaction();
		transaction.begin();
		
		//4. 进行持久化操作
		Customer customer = new Customer();
		customer.setAge(12);
		customer.setEmail("tom@atguigu.com");
		customer.setLastName("Tom");
		customer.setBirth(new Date());
		customer.setCreatedTime(new Date());
		
		//persist方法类似于Hibernate save方法,但是存在一个不同点:
		//在数据库主键自增情况下:若提前设置id属性值,JPA EntityManager 的 persist方法将抛出异常,而Hibernate Session的 save 方法可以执行,将忽略提前设置的id值
		entityManager.persist(customer);
		
		//5. 提交事务
		transaction.commit();
		
		//6. 关闭 EntityManager
		entityManager.close();
		
		//7. 关闭 EntityManagerFactory
		entityManagerFactory.close();
	}
}
5. 效果

打开数据库 , 发现数据库中自动创建了一张表和对应的字段 , 控制台输出了 hibernate 自动生成的 SQL 语句

JPA 注解介绍

在这里插入图片描述

@Entity

@Entity说明这个class是实体类,并且使用默认的orm规则,即class名对应数据库表中表名,class字段名即表中的字段名。
(如果想改变这种默认的orm规则,就要使用@Table来改变class名与数据库中表名的映射规则,@Column来改变class中字段名与db中表的字段名的映射规则)
元数据属性说明:

  • name: 表名

下面的代码说明,Customer类对应数据库中的Customer表,其中name为可选,缺省类名即表名!

    public class Customer { ... }
@Id

@Id 标注用于声明一个实体类的属性映射为数据库的主键列。该属性通常置于属性声明语句之前,可与声明语句同行,也可写在单独行上。
@Id标注也可置于属性的getter方法之前。

	//@Column: 指定数据库中对应的列名 , 如果数据库列名 和 实体字段名一样就可以省略不写
	@Column(name= "ID")
	//@GeneratedValue 用于标注主键的生成策略,通过strategy 属性指定。  GenerationType.AUTO: 系统自动创建
	@GeneratedValue(strategy=GenerationType.AUTO)
	//@Id 标注用于声明一个实体类的属性映射为数据库的主键列
	@Id
	public Integer getId() {
		return id;
	}
@Table

用来将实体类与数据库表映射起来 , 特别是类名与表民不同的情况

Table用来定义entity主表的name,catalog,schema等属性。
元数据属性说明:

  • name: 表名
  • catalog: 对应关系数据库中的catalog
  • schema:对应关系数据库中的schema
  • UniqueConstraints:定义一个UniqueConstraint数组,指定需要建唯一约束的列
    @Table(name="CUST")
    public class Customer { ... }
@Basic

如果自己不手动加 , 系统会默认已经加了@Basic注解
@Basic 表示一个简单的属性到数据库表的字段的映射,对于没有任何标注的 getXxxx() 方法,默认即为@Basic

  • fetch: 表示该属性的读取策略,有 EAGER 和 LAZY 两种,分别表示主支抓取和延迟加载,默认为 EAGER.
  • optional:表示该属性是否允许为null, 默认为true
@Column

实体的属性与其映射的数据库表的列不同名时需要使用@Column 标注说明
Column元数据定义了映射到数据库的列的所有属性:列名,是否唯一,是否允许为空,是否允许更新等。
元数据属性说明:

  • name:列名。
  • unique: 是否唯一
  • nullable: 是否允许为空
  • insertable: 是否允许插入
  • updatable: 是否允许更新
  • columnDefinition: 定义建表时创建此列的DDL
  • secondaryTable: 从表名。如果此列不建在主表上(默认建在主表),该属性定义该列所在从表的名字。
    public class Person {
    @Column(name = "PERSONNAME", unique = true, nullable = false, updatable = true)
    private String name;
 
    @Column(name = "PHOTO", columnDefinition = "BLOB NOT NULL", secondaryTable="PER_PHOTO")
    private byte[] picture;
@GeneratedValue

@GeneratedValue 用于标注主键的生成策略,通过 strategy 属性指定。默认情况下,JPA 自动选择一个最适合底层数据库的主键生成策略:SqlServer 对应 identity,MySQL 对应 auto increment。
在 javax.persistence.GenerationType 中定义了以下几种可供选择的策略:

  • IDENTITY:采用数据库 ID自增长的方式来自增主键字段,Oracle 不支持这种方式;
  • AUTO: JPA自动选择合适的策略,是默认选项;
  • SEQUENCE:通过序列产生主键,通过 @SequenceGenerator 注解指定序列名,MySql 不支持这种方式
  • TABLE:通过表产生主键,框架借由表模拟序列产生主键,使用该策略可以使应用更易于数据库移植。
@Temporal

默认生成的时间属性是datatime 类型 , 就是 年月日时分秒都有的
在这里插入图片描述

在核心的 Java API 中并没有定义 Date 类型的精度(temporal precision). 而在数据库中,表示 Date 类型的数据有 DATE, TIME, 和 TIMESTAMP 三种精度(即单纯的日期,时间,或者两者 兼备). 在进行属性映射时可使用@Temporal注解来调整精度.

    // TIMESTAMP : 年月日 时分秒
	@Temporal(TemporalType.TIMESTAMP)
	public Date getCreatedTime() {
		return createdTime;
	}
	//DATE : 年月日
	@Temporal(TemporalType.DATE)
	public Date getBirth() {
		return birth;
	}

@Transient

不需要为他生成数据表字段的时候
表示该属性并非一个到数据库表的字段的映射,ORM框架将忽略该属性.
如果一个属性并非数据库表的字段映射,就务必将其标示为@Transient,否则,ORM框架默认其注解为@Basic

	//工具方法. 不需要映射为数据表的一列. 
	@Transient
	public String getInfo(){
		return "lastName: " + lastName + ", email: " + email;
	}

在这里插入图片描述

JPA 的 API 接口

Persistence

Persistence 类是用于获取 EntityManagerFactory 实例。该类包含一个名为 createEntityManagerFactory 的 静态方法 。
createEntityManagerFactory 方法有如下两个重载版本。

  • 带有一个参数的方法以 JPA 配置文件 persistence.xml 中的持久化单元名为参数
  • 带有两个参数的方法:前一个参数含义相同,后一个参数 Map类型,用于设置 JPA 的相关属性,这时将忽略其它地方设置的属性。Map 对象的属性名必须是 JPA 实现库提供商的名字空间约定的属性名。
		//persistence 配置文件 <persistence-unit 标签中的 name 属性
		String persistenceUnitName = "jpa-1";
		
		Map<String, Object> properites = new HashMap<String, Object>();
		properites.put("hibernate.show_sql", true);
        EntityManagerFactory entityManagerFactory =
            Persistence.createEntityManagerFactory(persistenceUnitName);
        //properites是Map类型 , 用于设置JPA相关的属性
        EntityManagerFactory entityManagerFactory =
        	Persistence.createEntityManagerFactory(persistenceUnitName, properites);
EntityManagerFactory

EntityManagerFactory 接口主要用来创建 EntityManager 实例。该接口约定了如下4个方法:

  • createEntityManager():用于创建实体管理器对象实例。
  • createEntityManager(Map map):用于创建实体管理器对象实例的重载方法,Map 参数用于提供 EntityManager 的属性。
  • isOpen():检查 EntityManagerFactory 是否处于打开状态。实体管理器工厂创建后一直处于打开状态,除非调用close()方法将其关闭。
  • close():关闭 EntityManagerFactory 。 EntityManagerFactory 关闭后将释放所有资源,isOpen()方法测试将返回 false,其它方法将不能调用,否则将导致IllegalStateException异常。
		//2. 创建 EntityManager实例. 类似于 Hibernate 的 SessionFactory
		//EntityManagerFactory 接口主要用来创建 EntityManager 实例。
		EntityManager entityManager = entityManagerFactory.createEntityManager();
		
		//7. 关闭 EntityManagerFactory
		entityManagerFactory.close();        
EntityManager

在 JPA 规范中, EntityManager 是完成持久化操作的核心对象。实体作为普通 Java 对象,只有在调用 EntityManager 将其持久化后才会变成持久化对象。EntityManager 对象在一组实体类与底层数据源之间进行 O/R 映射的管理。它可以用来管理和更新 Entity Bean, 根椐主键查找 Entity Bean, 还可以通过JPQL语句查询实体。
实体的状态:

  • 新建状态: 新创建的对象,尚未拥有持久性主键。
  • 持久化状态:已经拥有持久性主键并和持久化建立了上下文环境
  • 游离状态:拥有持久化主键,但是没有与持久化建立上下文环境
  • 删除状态: 拥有持久化主键,已经和持久化建立上下文环境,但是从数据库中删除。
EntityManager#find方法

select查询
执行select语句 , 根据主键值 , 查找对应的对象类型的表中的数据

SELECT * FROM 表名 where id = ""

find (Class entityClass,Object primaryKey):返回指定的 OID 对应的实体类对象,如果这个实体存在于当前的持久化环境,则返回一个被缓存的对象;否则会创建一个新的 Entity, 并加载数据库中相关信息;若 OID 不存在于数据库中,则返回一个 null。第一个参数为被查询的实体类类型,第二个参数为待查找实体的主键值。

	@Test
	public void testFind() {
		Customer customer = entityManager.find(Customer.class, 1);
		System.out.println("-------------------------------------");
		
		System.out.println(customer);
	}

!!! 在调用find方法的时候 , 使用customer 之前就已经执行了sql语句
在这里插入图片描述

EntityManager#getReference

select查询
若实体Bean不存在,则抛出javax.persistence.EntityNotFoundException,另,不保 证 实体Bean 已被初始化

getReference (Class entityClass,Object primaryKey):与find()方法类似,不同的是:如果缓存中不存在指定的 Entity, EntityManager 会创建一个 Entity 类的代理对象,但是不会立即加载数据库中的信息,只有第一次真正使用此 Entity 的属性才加载,所以如果此 OID 在数据库不存在,getReference() 不会返回 null 值, 而是抛出EntityNotFoundException

代理对象可能会出现懒加载异常的问题

	@Test
	public void testGetReference(){
		Customer customer = entityManager.getReference(Customer.class, 1);
		System.out.println("-------------------------------------");
		System.out.println(customer);
	}

!!! 一开始调用getReference 方法只是返回了一个customer 的代理 , 只有当实际使用到customer 对象的时候才进行初始化, 才去执行sql语句.
在这里插入图片描述

EntityManager#persist

save存储 , 将对象持久化到数据库中 ,不能写ID

persist (Object entity):用于将新创建的 Entity 纳入到 EntityManager 的管理。该方法执行后,传入 persist() 方法的 Entity 对象转换成持久化状态。

  • 如果传入 persist() 方法的 Entity 对象已经处于持久化状态,则 persist() 方法什么都不做。
  • 如果对删除状态的 Entity 进行 persist() 操作,会转换为持久化状态。
  • 如果对游离状态的实体执行 persist() 操作,可能会在 persist() 方法抛出 EntityExistException(也有可能是在flush或事务提交后抛出)。
	//类似于 hibernate 的 save 方法. 使对象由临时状态变为持久化状态. 
	//和 hibernate 的 save 方法的不同之处: 若对象有 id, 则不能执行 insert 操作, 而会抛出异常. 
	@Test
	public void testPersistence(){
		Customer customer = new Customer();
		customer.setAge(15);
		customer.setBirth(new Date());
		customer.setCreatedTime(new Date());
		customer.setEmail("bb@163.com");
		customer.setLastName("BB");
		customer.setId(100);
		
		entityManager.persist(customer);
		System.out.println(customer.getId());
	}
EntityManager#remove

delete 根据 id 删除数据 , 不能是游离对象

remove (Object entity):删除实例。如果实例是被管理的,即与数据库实体记录关联,则同时会删除关联的数据库记录。

	//类似于 hibernate 中 Session 的 delete 方法. 把对象对应的记录从数据库中移除
	//但注意: 该方法只能移除 持久化 对象. 而 hibernate 的 delete 方法实际上还可以移除 游离对象.
	@Test
	public void testRemove(){
//		Customer customer = new Customer();
//		customer.setId(2);  //游离对象
		
		Customer customer = entityManager.find(Customer.class, 2);
		entityManager.remove(customer);
	}

!!!删除的时候 customer 必须包含全部数据 , 不能只有id

EntityManager#merge

merge (T entity):merge() 用于处理 Entity 的同步。即数据库的插入和更新操作
在这里插入图片描述

	/**
	 * 总的来说: 类似于 hibernate Session 的 saveOrUpdate 方法.
	 */
	//1. 若传入的是一个临时对象
	//会创建一个新的对象, 把临时对象的属性复制到新的对象中, 然后对新的对象执行持久化操作. 所以
	//新的对象中有 id, 但以前的临时对象中没有 id. 
	@Test
	public void testMerge1(){
		Customer customer = new Customer();
		customer.setAge(18);
		customer.setBirth(new Date());
		customer.setCreatedTime(new Date());
		customer.setEmail("cc@163.com");
		customer.setLastName("CC");
		
		Customer customer2 = entityManager.merge(customer);   //持久化新的对象
		
		System.out.println("customer#id:" + customer.getId());  //null
		System.out.println("customer2#id:" + customer2.getId()); 
	}
	//若传入的是一个游离对象, 即传入的对象有 OID. 
	//1. 若在 EntityManager 缓存中没有该对象
	//2. 若在数据库中也没有对应的记录
	//3. JPA 会创建一个新的对象, 然后把当前游离对象的属性复制到新创建的对象中
	//4. 对新创建的对象执行 insert 操作. 
	@Test
	public void testMerge2(){
		Customer customer = new Customer();
		customer.setAge(18);
		customer.setBirth(new Date());
		customer.setCreatedTime(new Date());
		customer.setEmail("dd@163.com");
		customer.setLastName("DD");
		
		customer.setId(100);
		
		Customer customer2 = entityManager.merge(customer);
		
		System.out.println("customer#id:" + customer.getId());
		System.out.println("customer2#id:" + customer2.getId());
	}
	//若传入的是一个游离对象, 即传入的对象有 OID. 
	//1. 若在 EntityManager 缓存中没有该对象
	//2. 若在数据库中也有对应的记录
	//3. JPA 会查询对应的记录, 然后返回该记录对一个的对象, 再然后会把游离对象的属性复制到查询到的对象中.
	//4. 对查询到的对象执行 update 操作. 
	@Test
	public void testMerge3(){
		Customer customer = new Customer();
		customer.setAge(18);
		customer.setBirth(new Date());
		customer.setCreatedTime(new Date());
		customer.setEmail("ee@163.com");
		customer.setLastName("EE");
		
		customer.setId(4);
		
		Customer customer2 = entityManager.merge(customer);
		
		System.out.println(customer == customer2); //false
	}
	//若传入的是一个游离对象, 即传入的对象有 OID. 
	//1. 若在 EntityManager 缓存中有对应的对象
	//2. JPA 会把游离对象的属性复制到查询到EntityManager 缓存中的对象中.
	//3. EntityManager 缓存中的对象执行 UPDATE. 
	@Test
	public void testMerge4(){
		Customer customer = new Customer();
		customer.setAge(18);
		customer.setBirth(new Date());
		customer.setCreatedTime(new Date());
		customer.setEmail("dd@163.com");
		customer.setLastName("DD");
		
		customer.setId(4);
		Customer customer2 = entityManager.find(Customer.class, 4);
		
		entityManager.merge(customer);
		
		System.out.println(customer == customer2); //false
	}
EntityManager# flush

强制发送sql语句 , 使数据表中的数据与 内存中的数据保持一致

  • 执行persist()、merger()时,数据并不是立即写入数据库中,而是由JPA缓存起来,在执行flush()时写入。在事务提交的时候,JPA会自动执行flush()一次性保存所有数据。如果需要立即保存,可手动执行flush()。

flush ():同步持久上下文环境,即将持久上下文环境的所有未保存实体的状态信息保存到数据库中。
setFlushMode (FlushModeType flushMode):设置持久上下文环境的Flush模式。参数可以取2个枚举

  • FlushModeType.AUTO 为自动更新数据库实体,
  • FlushModeType.COMMIT 为直到提交事务时才更新数据库记录。

getFlushMode ():获取持久上下文环境的Flush模式。返回FlushModeType类的枚举值。

	/**
	 * 同 hibernate 中 Session 的 flush 方法. 
	 */
	@Test
	public void testFlush(){
		Customer customer = entityManager.find(Customer.class, 1);
		System.out.println(customer);
		customer.setLastName("AA");
		entityManager.flush();
	}

解释:

  • 如果没有flush , 那么所有的sql 操作都要等 提交事务 commit之后再执行 , 然后数据库跟着改变

  • 如果有flush , 那么在调用flush之后就会执行sql语句 , 内存中数据已经改变了 , 但是 因为没有提交事务 , 所以数据库中的数据还没有改变 , 必须等commit提交之后数据库才会进行改变

  • refresh (Object entity):用数据库实体记录的值更新实体对象的状态,即更新实例的属性值。

  • clear ():清除持久上下文环境,断开所有关联的实体。如果这时还有未提交的更新则会被撤消。

  • contains (Object entity):判断一个实例是否属于当前持久上下文环境管理的实体。

  • isOpen ():判断当前的实体管理器是否是打开状态。

  • getTransaction ():返回资源层的事务对象。EntityTransaction实例可以用于开始和提交多个事务。

  • close ():关闭实体管理器。之后若调用实体管理器实例的方法或其派生的查询对象的方法都将抛

  • llegalstateException 异常,除了getTransaction 和 isOpen方法(返回 false)。不过,当与实体管理器关联的事务处于活动状态时,调用 close 方法后持久上下文将仍处于被管理状态,直到事务完成。

  • createQuery (String qlString):创建一个查询对象。

  • createNamedQuery (String name):根据命名的查询语句块创建查询对象。参数为命名的查询语句。

  • createNativeQuery (String sqlString):使用标准 SQL语句创建查询对象。参数为标准SQL语句字符串。

  • createNativeQuery (String sqls, String resultSetMapping):使用标准SQL语句创建查询对象,并指定返回结果集 Map的 名称。

EntityTransaction

EntityTransaction 接口用来管理资源层实体管理器的事务操作。通过调用实体管理器的getTransaction方法 获得其实例。

  • begin ()
    用于启动一个事务,此后的多个数据库操作将作为整体被提交或撤消。若这时事务已启动则会抛出 IllegalStateException 异常。
  • commit ()
    用于提交当前事务。即将事务启动以后的所有数据库更新操作持久化至数据库中。
  • rollback ()
    撤消(回滚)当前事务。即撤消事务启动后的所有数据库更新操作,从而不对数据库产生影响。
  • setRollbackOnly ()
    使当前事务只能被撤消。
  • getRollbackOnly ()
    查看当前事务是否设置了只能撤消标志。
  • isActive ()
    查看当前事务是否是活动的。如果返回true则不能调用begin方法,否则将抛出 IllegalStateException 异常;如果返回 false 则不能调用 commit、rollback、setRollbackOnly 及 getRollbackOnly 方法,否则将抛出 IllegalStateException 异常。

注解和接口总结:

注解作用
@Entity标注实体
@Id标注主键
@Table实体类与数据表映射
@Basic实体属性与数据表字段映射
@Column实体属性与数据表字段不一样的时候
@GeneratedValue主键的生成策略
@TemporaldataTime类型 , 年月日时分秒
Transient不需要生成数据表字段
API 接口与方法作用
Persistence用来获取 EntityManagerFactory 实例
EntityManagerFactory用来创建 EntityManager 实例
findselect语句 ,根据主键查询数据 ,返回实体类对象 , 先执行sql
getReferenceselect语句 ,根据主键查询数据 ,返回代理对象 , 使用后再执行sql
persistinsert语句, 存储对象进数据库 , 不能写ID
removedelete 根据 id 删除数据 , 不能是游离对象

映射关联关系

1.映射单向多对一的关联关系

举例: 订单与客户的关系 , 一个客户有多个订单 , 但是一个订单只能对应一个客户

  • 映射单向 n-1 的关联关系
  • 使用 @ManyToOne 来映射多对一的关联关系
  • 使用 @JoinColumn 来映射外键.
  • 可使用 @ManyToOne 的 fetch 属性来修改默认的关联属性的加载策略
  • 保存多对一时, 建议先保存 1 的一端, 后保存 n 的一端, 这样不会多出额外的 UPDATE 语句.
  • 默认情况下, 使用左外连接的方式来获取 n 的一端的对象和其关联的 1 的一端的对象.
  • 不能直接删除 1 的一端, 因为有外键约束. 会报错
@Table(name="JPA_ORDERS")		//对应的数据表名
@Entity		//表明是实体
public class Order {

	private Integer id;
	private String orderName;
    //订单对应的用户
	private Customer customer;

	@GeneratedValue		//生成主键的策略 : 默认
	@Id		//主键
	public Integer getId() {
		return id;
	}

	public void setId(Integer id) {
		this.id = id;
	}

	@Column(name="ORDER_NAME")		//列名与字段不同的情况
	public String getOrderName() {
		return orderName;
	}

	public void setOrderName(String orderName) {
		this.orderName = orderName;
	}

	//映射单向 n-1 的关联关系
	//使用 @ManyToOne 来映射多对一的关联关系
	//使用 @JoinColumn 来映射外键. 
	//可使用 @ManyToOne 的 fetch 属性来修改默认的关联属性的加载策略
	@JoinColumn(name="CUSTOMER_ID")
	@ManyToOne(fetch=FetchType.LAZY) 	//LAZY表示懒加载
	public Customer getCustomer() {
		return customer;
	}

	public void setCustomer(Customer customer) {
		this.customer = customer;
	}

}

插入数据 测试
如果先保存 1 的一端 , 就只会执行 3 遍 插入语句
如果先保存 多 的一端 , 就会执行 3遍插入 , 再执行一遍更新

	/**
	 * 保存多对一时, 建议先保存 1 的一端, 后保存 n 的一端, 这样不会多出额外的 UPDATE 语句.
	 */
	@Test
	public void testManyToOnePersist(){
		Customer customer = new Customer();
		customer.setAge(18);
		customer.setBirth(new Date());
		customer.setCreatedTime(new Date());
		customer.setEmail("gg@163.com");
		customer.setLastName("GG");
		
		Order order1 = new Order();
		order1.setOrderName("G-GG-1");
		
		Order order2 = new Order();
		order2.setOrderName("G-GG-2");
		
		//设置关联关系
		order1.setCustomer(customer);
		order2.setCustomer(customer);
		
		//执行保存操作
		entityManager.persist(customer); //1的一方
		entityManager.persist(order1);		//多的一方
		entityManager.persist(order2);		//多的一方
	}

取出数据 测试

	//默认情况下, 使用左外连接的方式来获取 n 的一端的对象和其关联的 1 的一端的对象. 
	//可使用 @ManyToOne 的 fetch 属性来修改默认的关联属性的加载策略
	//如果设置@ManyToOne(fetch=FetchType.LAZY)  懒加载策略 , 那么 会 先执行 order的sql然后输出 , 载执行 Customer的sql然后输出
	@Test
	public void testManyToOneFind(){
		Order order = entityManager.find(Order.class, 1);
		System.out.println(order.getOrderName());
		
		System.out.println(order.getCustomer().getLastName());
	}

删除数据 测试

	//不能直接删除 1 的一端, 因为有外键约束.  会报错
	//先删除 多 的一端  , 然后删除 1 的一端
	@Test
	public void testManyToOneRemove(){
//		Order order = entityManager.find(Order.class, 1);
//		entityManager.remove(order);

		Customer customer = entityManager.find(Customer.class, 7);
		entityManager.remove(customer);		//失败 , 外键约束
	}

2.映射单向一对多的关联关系

  • 使用 @OneToMany 来映射 1-n 的关联关系 , ,cascade 属性来修改默认的删除策略.
  • 使用 @JoinColumn 来映射外键列的名称, fetch 属性来修改默认的加载策略
  • 默认对关联的多的一方使用懒加载的加载策略.
  • 单向 1-n 关联关系执行保存时, 一定会多出 UPDATE 语句. 因为 n 的一端在插入时不会同时插入外键列.
  • 默认情况下, 若删除 1 的一端, 则会先把关联的 n 的一端的外键置空 (变为null), 然后进行删除. 可以通过 @OneToMany 的 cascade 属性来修改默认的删除策略. @OneToMany(cascade={CascadeType.REMOVE})表示在删除 1 的一端的时候 同步删除 多 的一端 , 也就是级联删除
public class Customer {
	private Set<Order> orders = new HashSet<>();

	//映射单向 1-n 的关联关系
	//使用 @OneToMany 来映射 1-n 的关联关系
	//使用 @JoinColumn 来映射外键列的名称
	//可以使用 @OneToMany 的 fetch 属性来修改默认的加载策略
	//可以通过 @OneToMany 的 cascade 属性来修改默认的删除策略. 
	//注意: 若在 1 的一端的 @OneToMany 中使用 mappedBy 属性, 则 @OneToMany 端就不能再使用 @JoinColumn 属性了. 
//	@JoinColumn(name="CUSTOMER_ID")
	@OneToMany(fetch=FetchType.LAZY,cascade={CascadeType.REMOVE},mappedBy="customer")
	public Set<Order> getOrders() {
		return orders;
	}

	public void setOrders(Set<Order> orders) {
		this.orders = orders;
	}
}

插入数据 测试

	//单向 1-n 关联关系执行保存时, 一定会多出 UPDATE 语句.
	//因为 n 的一端在插入时不会同时插入外键列. 
	@Test
	public void testOneToManyPersist(){
		Customer customer = new Customer();
		customer.setAge(18);
		customer.setBirth(new Date());
		customer.setCreatedTime(new Date());
		customer.setEmail("mm@163.com");
		customer.setLastName("MM");
		
		Order order1 = new Order();
		order1.setOrderName("O-MM-1");
		
		Order order2 = new Order();
		order2.setOrderName("O-MM-2");
		
		//建立关联关系
		customer.getOrders().add(order1);
		customer.getOrders().add(order2);
		
		order1.setCustomer(customer);
		order2.setCustomer(customer);
		
		//执行保存操作
		entityManager.persist(customer);

		entityManager.persist(order1);
		entityManager.persist(order2);
	}

删除数据 测试

	//默认情况下, 若删除 1 的一端, 则会先把关联的 n 的一端的外键置空 (变为null), 然后进行删除. 
	//可以通过 @OneToMany 的 cascade 属性来修改默认的删除策略. 
	//cascade={CascadeType.REMOVE} 表示在删除 1 的一端的时候 同步删除 多 的一端
	@Test
	public void testOneToManyRemove(){
		Customer customer = entityManager.find(Customer.class, 8);
		entityManager.remove(customer);		//删除customer的时候 , order的外键会置空
	}

修改数据 测试

	@Test
	public void testUpdate(){
		Customer customer = entityManager.find(Customer.class, 10);
		
		customer.getOrders().iterator().next().setOrderName("O-XXX-10");
	}

3.映射双向多对一的关联关系

双向一对多关系中,必须存在一个关系维护端,在 JPA 规范中,要求 many 的一方作为关系的维护端(owner side), one 的一方作为被维护端(inverse side)。
可以在 one 方指定 @OneToMany 注释并设置 mappedBy 属性,以指定它是这一关联中的被维护端,many 为维护端。
在 many 方指定 @ManyToOne 注释,并使用 @JoinColumn 指定外键名称

//1 的一方
	private Set<Order> orders = new HashSet<>();
	
    //	@JoinColumn(name="CUSTOMER_ID")
	@OneToMany(fetch=FetchType.LAZY,cascade={CascadeType.REMOVE},mappedBy="customer")
	public Set<Order> getOrders() {
		return orders;
	}
	public void setOrders(Set<Order> orders) {
		this.orders = orders;
	}


//多 的一方
	private Customer customer;

	@JoinColumn(name="CUSTOMER_ID")
	@ManyToOne(fetch=FetchType.LAZY)
	public Customer getCustomer() {
		return customer;
	}

	public void setCustomer(Customer customer) {
		this.customer = customer;
	}
	//若是双向 1-n 的关联关系, 执行保存时
	//若先保存 n 的一端, 再保存 1 的一端, 默认情况下, 会多出 n 条 UPDATE 语句.
	//若先保存 1 的一端, 则会多出 n/2 条 UPDATE 语句
	
	//建议 : 在进行双向 1-n 关联关系时, 建议使用 n 的一方来维护关联关系, 而 1 的一方不维护关联系, 这样会有效的减少 SQL 语句. 
	//注意: 若在 1 的一端的 @OneToMany 中使用 mappedBy 属性, 则 @OneToMany 端就不能再使用 @JoinColumn 属性了. 

4.映射双向一对一的关联关系

比如: 部门与经理 一个部门只有一个经理 一个经理只管一个部门
基于外键的 1-1 关联关系:在双向的一对一关联中,需要在关系被维护端(inverse side)中的 @OneToOne 注释中指定 mappedBy,以指定是这一关联中的被维护端。同时需要在关系维护端(owner side)建立外键列指向关系被维护端的主键列。

如果延迟加载要起作用, 就必须设置一个代理对象.
Manager 其实可以不关联一个 Department
如果有 Department 关联就设置为代理对象而延迟加载, 如果不存在关联的 Department 就设置 null, 因为外键字段是定义在 Department 表中的,Hibernate 在不读取 Department 表的情况是无法判断是否有关联有 Deparmtment, 因此无法判断设置 null 还是代理对象, 而统一设置为代理对象,也无法满足不关联的情况, 所以无法使用延迟加载,只 有显式读取 Department.

@Table(name="JPA_DEPARTMENTS")
@Entity
public class Department {

	private Integer id;
	private String deptName;
	
	private Manager mgr;
	//使用 @OneToOne 来映射 1-1 关联关系。
	//若需要在当前数据表中添加外键则需要使用 @JoinColumn 来进行映射. 注意, 1-1 关联关系, 所以需要添加 unique=true
	@JoinColumn(name="MGR_ID", unique=true)
	@OneToOne(fetch=FetchType.LAZY)
	public Manager getMgr() {
		return mgr;
	}

	public void setMgr(Manager mgr) {
		this.mgr = mgr;
	}
}
@Table(name="JPA_MANAGERS")
@Entity
public class Manager {

	private Integer id;
	private String mgrName;
	
	private Department dept;
	//对于不维护关联关系, 没有外键的一方, 使用 @OneToOne 来进行映射, 建议设置 mappedBy=true
	@OneToOne(mappedBy="mgr")
	public Department getDept() {
		return dept;
	}

	public void setDept(Department dept) {
		this.dept = dept;
	}
}

添加数据 测试

	//双向 1-1 的关联关系, 建议先保存不维护关联关系的一方, 即没有外键的一方, 这样不会多出 UPDATE 语句.
	// 如果先保存 有 外键的一方 会多出 UPDATE 语句
	@Test
	public void testOneToOnePersistence(){
		Manager mgr = new Manager();
		mgr.setMgrName("M-BB");
		
		Department dept = new Department();
		dept.setDeptName("D-BB");
		
		//设置关联关系
		mgr.setDept(dept);
		dept.setMgr(mgr);
		
		//执行保存操作
		entityManager.persist(mgr);
		entityManager.persist(dept);
	}

获取数据 测试

	//1.默认情况下, 若获取维护关联关系的一方(有外键的一方), 则会通过左外连接获取其关联的对象. 
	//但可以通过 @OntToOne 的 fetch 属性来修改加载策略.
	@Test
	public void testOneToOneFind(){
		Department dept = entityManager.find(Department.class, 1);
		System.out.println(dept.getDeptName());
		System.out.println(dept.getMgr().getClass().getName());
	}

5.映射双向多对多的关联关系

在双向多对多关系中,我们必须指定一个关系维护端(owner side),可以通过 @ManyToMany 注释中指定 mappedBy 属性来标识其为关系维护端。

双向多对多 必须 有一方 放弃维护关联关系 , 否则就会出现主键重复的问题

比如: 商品与类别 , 一个类别有多个商品 一个商品可以属于多个类别

商品类: Item 类别类: Category
类别

@Table(name="JPA_CATEGORIES")
@Entity
public class Category {

	private Integer id;
	private String categoryName;
	
	private Set<Item> items = new HashSet<>();		//商品可以有多个类别
	
	@ManyToMany(mappedBy="categories")
	public Set<Item> getItems() {
		return items;
	}

	public void setItems(Set<Item> items) {
		this.items = items;
	}
}

商品

@Table(name="JPA_ITEMS")
@Entity
public class Item {

	private Integer id;
	private String itemName;
	
	private Set<Category> categories = new HashSet<>();		//类别可以有多个商品


	//使用 @ManyToMany 注解来映射多对多关联关系
	//使用 @JoinTable 来映射中间表
	//1. name 指向中间表的名字
	//2. joinColumns 映射当前类所在的表在中间表中的外键
	//2.1 name 指定外键列的列名
	//2.2 referencedColumnName 指定外键列关联当前表的哪一列
	//3. inverseJoinColumns 映射关联的类所在中间表的外键
	@JoinTable(name="ITEM_CATEGORY",
			joinColumns={@JoinColumn(name="ITEM_ID", referencedColumnName="ID")},
			inverseJoinColumns={@JoinColumn(name="CATEGORY_ID", referencedColumnName="ID")})
	@ManyToMany
	public Set<Category> getCategories() {
		return categories;
	}

	public void setCategories(Set<Category> categories) {
		this.categories = categories;
	}
}

保存数据 测试

	//多对所的保存
	@Test
	public void testManyToManyPersist(){
		Item i1 = new Item();
		i1.setItemName("i-1");
	
		Item i2 = new Item();
		i2.setItemName("i-2");
		
		Category c1 = new Category();
		c1.setCategoryName("C-1");
		
		Category c2 = new Category();
		c2.setCategoryName("C-2");
		
		//设置关联关系
		i1.getCategories().add(c1);
		i1.getCategories().add(c2);
		
		i2.getCategories().add(c1);
		i2.getCategories().add(c2);
		
		c1.getItems().add(i1);
		c1.getItems().add(i2);
		
		c2.getItems().add(i1);
		c2.getItems().add(i2);
		
		//执行保存  8条 insert 语句
		entityManager.persist(i1);
		entityManager.persist(i2);
		entityManager.persist(c1);
		entityManager.persist(c2);
	}

获取数据 测试

	//对于关联的集合对象, 默认使用懒加载的策略.
	//使用维护关联关系的一方获取, 还是使用不维护关联关系的一方获取, SQL 语句相同. 
	//不管使用有外键的一方 还是 使用没有外键的一方 最后 执行的 sql 语句是相同的
	@Test
	public void testManyToManyFind(){
//		Item item = entityManager.find(Item.class, 5);
//		System.out.println(item.getItemName());
//		
//		System.out.println(item.getCategories().size());
		
		Category category = entityManager.find(Category.class, 3);
		System.out.println(category.getCategoryName());
		System.out.println(category.getItems().size());
	}

使用二级缓存

二级缓存的作用是什么

二级缓存的目的: 可以跨 entityManager , 关闭与开启都不会影响 耳机缓存的数据 , 就只用发 一条 sql 语句

shared-cache-mode节点:若 JPA 实现支持二级缓存,该节点可以配置在当前的持久化单元中是否启用二级缓存,可配置如下值:

  • ALL:所有的实体类都被缓存
  • NONE:所有的实体类都不被缓存.
  • ENABLE_SELECTIVE:标识 @Cacheable(true) 注解的实体类将被缓存
  • DISABLE_SELECTIVE:缓存除标识 @Cacheable(false) 以外的所有实体类
  • UNSPECIFIED:默认值,JPA 产品默认值将被使用
    //只会执行一条 sql 使用了一级缓存 , 缓存中有customer1 执行过的记录
	@Test
	public void testSecondLevelCache(){
		Customer customer1 = entityManager.find(Customer.class, 1);
		Customer customer2 = entityManager.find(Customer.class, 1);
	}
    // 会执行两条 sql  , 因为 entityManager 关闭 再 开启 , 一级缓存中的数据就被清除了
	@Test
	public void testSecondLevelCache(){
		Customer customer1 = entityManager.find(Customer.class, 1);
		
		transaction.commit();
		entityManager.close();
		
		entityManager = entityManagerFactory.createEntityManager();
		transaction = entityManager.getTransaction();
		transaction.begin();
		
		Customer customer2 = entityManager.find(Customer.class, 1);
	}

配置二级缓存
在persistence.xml中添加以下代码 ,此外还需要导入 依赖包 ehcache

		<!-- 添加持久化类  , 将需要持久化的实体类配置在这里 , 只有将实体类加在这里 , JPA才能对实体类进行操作-->
		<class>com.atguigu.jpa.helloworld.Customer</class>
		<class>com.atguigu.jpa.helloworld.Order</class>
	    
	    <!--放在添加持久化类的下面-->
		<!--
		配置二级缓存的策略
		ALL:所有的实体类都被缓存
		NONE:所有的实体类都不被缓存.
		ENABLE_SELECTIVE:标识 @Cacheable(true) 注解的实体类将被缓存
		DISABLE_SELECTIVE:缓存除标识 @Cacheable(false) 以外的所有实体类
		UNSPECIFIED:默认值,JPA 产品默认值将被使用
		-->
		<shared-cache-mode>ENABLE_SELECTIVE</shared-cache-mode>


        <properties>
			<!-- 二级缓存相关 -->
<!--			启用二级缓存-->
			<property name="hibernate.cache.use_second_level_cache" value="true"/>
<!--			配置二级缓存的是此方案产品   value是包的名字-->
			<property name="hibernate.cache.region.factory_class" value="org.hibernate.cache.ehcache.EhCacheRegionFactory"/>
<!--			查询缓存-->
			<property name="hibernate.cache.use_query_cache" value="true"/>
		</properties>	

然后在要缓存的类上加上注解: @Cacheable(true) 表示这个类开启二级缓存

@Cacheable(true)
//数据库中要创建的对应的表名
@Table(name="JPA_CUTOMERS")
//告诉JPA这是一个需要持久化的类
@Entity
public class Customer {

}

JPQL

  • JPQL语言,即 Java Persistence Query Language 的简称。JPQL 是一种和 SQL 非常类似的中间性和对象化查询语言,它最终会被编译成针对不同底层数据库的 SQL 查询,从而屏蔽不同数据库的差异。
  • JPQL语言的语句可以是 select 语句、update 语句或delete语句,它们都通过 Query 接口封装执行
  • Query接口封装了执行数据库查询的相关方法。调用 EntityManagercreateQuery、create NamedQuery 及 createNativeQuery 方法可以获得查询对象,进而可调用 Query 接口的相关方法来执行查询操作。

使用范围

  • createQuery是做原始的使用方式
  • createNamedQuery 适用于在实体类前使用 @NamedQuery 标记的查询语句
  • createNativeQuery 适用于本地 SQL

NamedQuery
在类上加注解NamedQuery

@NamedQuery(name="testNamedQuery", query="FROM Customer c WHERE c.id = ?")
@Cacheable(true)
//数据库中要创建的对应的表名
@Table(name="JPA_CUTOMERS")
//告诉JPA这是一个需要持久化的类
@Entity
public class Customer {

NamedQuery测试

	//createNamedQuery 适用于在实体类前使用 @NamedQuery 标记的查询语句
	@Test
	public void testNamedQuery(){
		Query query = entityManager.createNamedQuery("testNamedQuery").setParameter(1, 3);
		Customer customer = (Customer) query.getSingleResult();

		System.out.println(customer);
	}

createNativeQuery 测试

	//createNativeQuery 适用于本地 SQL
	@Test
	public void testNativeQuery(){
		String sql = "SELECT age FROM jpa_cutomers WHERE id = ?";
		Query query = entityManager.createNativeQuery(sql).setParameter(1, 3);
		
		Object result = query.getSingleResult();
		System.out.println(result);
	}
Query接口的主要方法
  • int executeUpdate()
    用于执行update或delete语句。
  • List getResultList()
    用于执行select语句并返回结果集实体列表。获取集合对象
  • Object getSingleResult()
    用于执行只返回单个结果实体的select语句。 获取单个值
  • Query setFirstResult(int startPosition)
    用于设置从哪个实体记录开始返回查询结果。
  • Query setMaxResults(int maxResult)
    用于设置返回结果实体的最大数。与setFirstResult结合使用可实现分页查询
  • Query setFlushMode(FlushModeType flushMode)
    设置查询对象的Flush模式。参数可以取2个枚举值:FlushModeType.AUTO 为自动更新数据库记录,FlushMode Type.COMMIT 为直到提交事务时才更新数据库记录。
  • setHint(String hintName, Object value)
    设置与查询对象相关的特定供应商参数或提示信息。参数名及其取值需要参考特定 JPA 实现库提供商的文档。如果第二个参数无效将抛出IllegalArgumentException异常。 一般用来设置启用查询缓存
  • setParameter(int position, Object value)
    为查询语句的指定位置参数赋值。Position 指定参数序号,value 为赋给参数的值。
  • setParameter(int position, Date d, TemporalType type)
    为查询语句的指定位置参数赋 Date 值。Position 指定参数序号,value 为赋给参数的值,temporalType 取 TemporalType 的枚举常量,包括 DATE、TIME 及 TIMESTAMP 三个,,用于将 Java 的 Date 型值临时转换为数据库支持的日期时间类型(java.sql.Date、java.sql.Time及java.sql.Timestamp)。
  • setParameter(int position, Calendar c, TemporalType type)
    为查询语句的指定位置参数赋 Calenda r值。position 指定参数序号,value 为赋给参数的值,temporalType 的含义及取舍同前。
  • setParameter(String name, Object value)
    为查询语句的指定名称参数赋值。
  • setParameter(String name, Date d, TemporalType type)
    为查询语句的指定名称参数赋 Date 值。用法同前。
  • setParameter(String name, Calendar c, TemporalType type)
    为查询语句的指定名称参数设置Calendar值。name为参数名,其它同前。该方法调用时如果参数位置或参数名不正确,或者所赋的参数值类型不匹配,将抛出 IllegalArgumentException 异常。
语句用法

select语句用于执行查询。其语法可表示为:
select_clause
form_clause
[where_clause]
[groupby_clause]
[having_clause]
[orderby_clause]

select-from 子句

  • from 子句是查询语句的必选子句。
    Select 用来指定查询返回的结果实体或实体的某些属性
    From 子句声明查询源实体类,并指定标识符变量(相当于SQL表的别名)。
  • 如果不希望返回重复实体,可使用关键字 distinct 修饰。select、from 都是 JPQL 的关键字,通常全大写或全小写,建议不要大小写混用。

查询所有实体

  • 查询所有实体的 JPQL 查询字串很简单,例如:
    select o from Order o 或 select o from Order as o
  • 关键字 as 可以省去。
  • 标识符变量的命名规范与 Java 标识符相同,且区分大小写。

调用 EntityManager 的 createQuery() 方法可创建查询对象,接着调用 Query 接口的 getResultList() 方法就可获得查询结果集。例如:

Query query = entityManager.createQuery( "select o from Order o"); 
List orders = query.getResultList();
Iterator iterator = orders.iterator();
while( iterator.hasNext() ) {
	// 处理Order
}

where子句

  • where子句用于指定查询条件,where跟条件表达式。例:
    select o from Orders o where o.id = 1
    select o from Orders o where o.id > 3 and o.confirm = ‘true’
    select o from Orders o where o.address.streetNumber >= 123
  • JPQL也支持包含参数的查询,例如:
    select o from Orders o where o.id = :myId
    select o from Orders o where o.id = :myId and o.customer = :customerName
     注意:参数名前必须冠以冒号(😃,执行查询前须使用Query.setParameter(name, value)方法给参数赋值。

获取对象集合

	@Test
	public void testHelloJPQL(){
		String jpql = "FROM Customer c WHERE c.age > ?";	//要执行的语句
		Query query = entityManager.createQuery(jpql);	//创建createQuery 接口对象
		
		//占位符的索引是从 1 开始
		query.setParameter(1, 1);	//为查询语句的指定位置参数赋值。
		List<Customer> customers = query.getResultList();	//执行select语句并返回结果集
		System.out.println(customers.size());
	}

只查询部分属性

	//默认情况下, 若只查询部分属性, 则将返回 Object[] 类型的结果. 或者 Object[] 类型的 List.
	//如果希望返回的是对象 ,也可以在实体类中创建对应的构造器, 然后再 JPQL 语句中利用对应的构造器返回实体类的对象.
	@Test
	public void testPartlyProperties(){
	    //返回数组 , 数组中是 lastName 与 age  连个元素
		//String jpql = "SELECT c.lastName, c.age FROM Customer c WHERE c.id > ?";
		String jpql = "SELECT new Customer(c.lastName, c.age) FROM Customer c WHERE c.id > ?";
		List result = entityManager.createQuery(jpql).setParameter(1, 1).getResultList();
		
		System.out.println(result);
	}
使用 Hibernate 的查询缓存

前提 必须在 persistence.xml 配置文件中开启二级缓存

	//使用 hibernate 的查询缓存. 
	@Test
	public void testQueryCache(){
		String jpql = "FROM Customer c WHERE c.age > ?";
		//Query query = entityManager.createQuery(jpql);  最初的写法 , 默认是两条sql语句
		// 在后天添加  setHint(QueryHints.HINT_CACHEABLE, true) 表示开启缓存 , 就只会执行一条sql
		Query query = entityManager.createQuery(jpql).setHint(QueryHints.HINT_CACHEABLE, true);
		
		//占位符的索引是从 1 开始
		query.setParameter(1, 1);
		List<Customer> customers = query.getResultList();
		System.out.println(customers.size());
		
		//query = entityManager.createQuery(jpql);   最初的写法 , 默认是两条sql语句
		query = entityManager.createQuery(jpql).setHint(QueryHints.HINT_CACHEABLE, true);
		
		//占位符的索引是从 1 开始
		query.setParameter(1, 1);
		customers = query.getResultList();
		System.out.println(customers.size());
	}
order by子句

order by子句用于对查询结果集进行排序。和SQL的用法类似,可以用 “asc“ 和 "desc“ 指定升降序。如果不显式注明,默认为升序。
select o from Orders o order by o.id
select o from Orders o order by o.address.streetNumber desc
select o from Orders o order by o.customer asc, o.id desc

	@Test
	public void testOrderBy(){
		String jpql = "FROM Customer c WHERE c.age > ? ORDER BY c.age DESC";
		Query query = entityManager.createQuery(jpql).setHint(QueryHints.HINT_CACHEABLE, true);
		
		//占位符的索引是从 1 开始
		query.setParameter(1, 1);
		List<Customer> customers = query.getResultList();
		System.out.println(customers.size());
	}
group by子句与聚合查询 与 having子句
  • group by 子句用于对查询结果分组统计,通常需要使用聚合函数。常用的聚合函数主要有 AVG、SUM、COUNT、MAX、MIN 等,它们的含义与SQL相同。例如:
    select max(o.id) from Orders o
  • 没有 group by 子句的查询是基于整个实体类的,使用聚合函数将返回单个结果值,可以使用Query.getSingleResult()得到查询结果。例如:
    Query query = entityManager.createQuery(
    “select max(o.id) from Orders o”);
    Object result = query.getSingleResult();
    Long max = (Long)result;
    … …

having子句

  • Having 子句用于对 group by 分组设置约束条件,用法与where 子句基本相同,不同是 where 子句作用于基表或视图,以便从中选择满足条件的记录;having 子句则作用于分组,用于选择满足条件的组,其条件表达式中通常会使用聚合函数。
  • 例如,以下语句用于查询订购总数大于100的商家所售商品及数量:
    select o.seller, o.goodId, sum(o.amount) from V_Orders o group by
    o.seller, o.goodId having sum(o.amount) > 100
  • having子句与where子句一样都可以使用参数。
	//查询 order 数量大于 2 的那些 Customer
	@Test
	public void testGroupBy(){
		String jpql = "SELECT o.customer FROM Order o "
				+ "GROUP BY o.customer "
				+ "HAVING count(o.id) >= 2";
		List<Customer> customers = entityManager.createQuery(jpql).getResultList();
		
		System.out.println(customers);
	}
关联查询

在JPQL中,很多时候都是通过在实体类中配置实体关联的类属性来实现隐含的关联(join)查询。例如:
select o from Orders o where o.address.streetNumber=2000
上述JPQL语句编译成以下SQL时就会自动包含关联,默认为左关联。
在某些情况下可能仍然需要对关联做精确的控制。为此,JPQL 也支持和 SQL 中类似的关联语法。如:

left out join / left join
inner join
left join / inner join fetch

其中,left join和left out join等义,都是允许符合条件的右边表达式中的实体为空。

例如,

  • 以下外关联查询可以找出所有客户实体记录,即使它未曾订货:
    select c from Customers c left join c.orders o
  • 以下内关联查询只找出所有曾订过商品的客户实体记录:
    select c from Customers c inner join c.orders o
  • 如果001号客户下过5次订单的话,以下fetch关联查询将得到 5个客户实体的引用,并且执行了 5 个订单的查询:
    select c from Customers c left join fetch c.orders o where c.id=001
	/**
	 * JPQL 的关联查询同 HQL 的关联查询. 
	 */
	@Test
	public void testLeftOuterJoinFetch(){
	//因为customer与Orders关联了, 并且 下面又使用Orders , 所以不使用关联查询就会发送两条sql
	    // String jpql = "FROM Customer c  WHERE c.id = ?";
        
        //建议加上 FETCH  这样得到的就是一个对象 , 并且对象已经初始化成功了 
		String jpql = "FROM Customer c LEFT OUTER JOIN FETCH c.orders WHERE c.id = ?";
		
		Customer customer = 
				(Customer) entityManager.createQuery(jpql).setParameter(1, 12).getSingleResult();
		System.out.println(customer.getLastName());
		
		System.out.println(customer.getOrders().size());
		
		//如果不使用  FETCH  返回的可能是多条记录 ,每个记录都是一个对象数组 , 而使用getSingleResult接受单个对象 就会报错
//		List<Object[]> result = entityManager.createQuery(jpql).setParameter(1, 12).getResultList();
//		System.out.println(result);
	}
子查询

JPQL也支持子查询,在 where 或 having 子句中可以包含另一个查询。当子查询返回多于 1 个结果集时,它常出现在 any、all、exist s表达式中用于集合匹配查询。它们的用法与SQL语句基本相同。

	@Test
	public void testSubQuery(){
		//查询所有 Customer 的 lastName 为 YY 的 Order
		String jpql = "SELECT o FROM Order o "
				+ "WHERE o.customer = (SELECT c FROM Customer c WHERE c.lastName = ?)";
		
		Query query = entityManager.createQuery(jpql).setParameter(1, "YY");
		List<Order> orders = query.getResultList();
		System.out.println(orders.size());
	}
JPQL函数

JPQL提供了以下一些内建函数,包括字符串处理函数、算术函数和日期函数。
字符串处理函数主要有:

  • concat(String s1, String s2):字符串合并/连接函数。
  • substring(String s, int start, int length):取字串函数。
  • trim([leading|trailing|both,] [char c,] String s):从字符串中去掉首/尾指定的字符或空格。
  • lower(String s):将字符串转换成小写形式。
  • upper(String s):将字符串转换成大写形式。
  • length(String s):求字符串的长度。
  • locate(String s1, String s2[, int start]):从第一个字符串中查找第二个字符串(子串)出现的位置。若未找到则返回0。

算术函数主要有 abs、mod、sqrt、size 等。Size 用于求集合的元素个数。
日期函数主要为三个,即 current_date、current_time、current_timestamp,它们不需要参数,返回服务器上的当前日期、时间和时戳。

	//使用 jpql 内建的函数
	@Test
	public void testJpqlFunction(){
		String jpql = "SELECT lower(c.email) FROM Customer c";
		
		List<String> emails = entityManager.createQuery(jpql).getResultList();
		System.out.println(emails);
	}
update语句 与 delete语句
	//可以使用 JPQL 完成 UPDATE 和 DELETE 操作. 
	@Test
	public void testExecuteUpdate(){
		String jpql = "UPDATE Customer c SET c.lastName = ? WHERE c.id = ?";
		Query query = entityManager.createQuery(jpql).setParameter(1, "YYY").setParameter(2, 12);
		
		query.executeUpdate();
	}

Spring 整合 JPA

主要从两方面

  1. 让 spring 去管理 JPA 的 EntityManagerFactory
  2. 让 JPA 用上 spring 的声明式事务

三种整合方式:

  • LocalEntityManagerFactoryBean:适用于那些仅使用 JPA 进行数据访问的项目,该 FactoryBean 将根据JPA PersistenceProvider 自动检测配置文件进行工作,一般从“META-INF/persistence.xml”读取配置信息,这种方式最简单,但不能设置 Spring 中定义的DataSource,且不支持 Spring 管理的全局事务 (基本不用这种)
  • 从JNDI中获取:用于从 Java EE 服务器获取指定的EntityManagerFactory,这种方式在进行 Spring 事务管理时一般要使用 JTA 事务管理(必须要有JavaEE服务器, 所以基本也不使用)
  • LocalContainerEntityManagerFactoryBean:适用于所有环境的 FactoryBean,能全面控制 EntityManagerFactory 配置,如指定 Spring 定义的 DataSource 等等。

Spring Boot整合JPA

SpringBoot整合JPA

  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值