Spring Data JPA

Spring JPA

1、Spring Data

1.1、简介

 Spring Data’s mission is to provide a familiar and consistent, Spring-based programming model for data access while still retaining the special traits of the underlying data store.
 ​
 It makes it easy to use data access technologies, relational and non-relational databases, map-reduce frameworks, and cloud-based data services. This is an umbrella project which contains many subprojects that are specific to a given database. The projects are developed by working together with many of the companies and developers that are behind these exciting technologies.
 ​

Spring Data 的任务是为数据访问提供一个熟悉的、一致的、基于 Spring 的编程模型,同时保留底层数据存储的特性。

它使得使用数据访问技术、关系和非关系数据库、 map-reduce 框架和基于云的数据服务变得容易。这是一个总括项目,其中包含许多特定于给定数据库的子项目。这些项目是通过与这些令人兴奋的技术背后的许多公司和开发人员合作开发的。

简单来说,可以理解为:

Spring Data致力为数据访问(Dao)提供熟悉且一致的基于Spring的编程模板:

对于每种持久性存储,您的 Dao 通常需要为不同存储库提供不同 CRUD (创建-读取-更新-删除)持久化操作!Spring Data为这些持久化存储以及特定实现提供了通用接口(CrudRepository、PagingAndSortingRepository)和模板(jdbcTemplate、redisTemplate、RestTemplate、MongoTemplate….)!

其目的是统一和简化对不同类型持久化存储(关系型数据库和 NoSQL数据存储)的访问!

spring官网:spring.io

github:https://github.com/spring-projects

hibernate:https://hibernate.org/

1.2、Spring Data 主要模块

Spring Data 支持的持久层技术非常多,如下所示:

  • Spring Data Commons - 支撑每个 Spring 数据模块的核心 Spring 概念。

  • Spring Data JDBC - 对 JDBC 的 Spring Data 存储库支持。

  • Spring Data JDBC Ext - 支持特定于数据库的标准 JDBC 扩展,包括支持 Oracle RAC 快速连接故障转移、支持 AQ JMS 和支持使用高级数据类型。

  • Spring Data JPA - Spring Data 存储库对 JPA 的支持。

  • Spring Data KeyValue - Map基于映射的存储库和 SPI,可以轻松地为键值存储构建 Spring Data 模块。

  • Spring Data LDAP - SpringData 存储库对 SpringLDAP 的支持。

  • Spring Data MongoDB - MongoDB 基于 Spring 的对象文档支持和存储库。

  • Spring Data Redis - 从 Spring 应用程序轻松配置和访问 Redis。

  • Spring Data REST - 将 Spring Data 存储库作为超媒体驱动的 RESTful 资源导出。

  • Spring Data for Apache Cassandra - 易于配置和访问 Cassandra 或大规模、高可用性、面向数据的 Spring 应用程序。

  • Spring Data for Apache Geode - 为高度一致、低延迟、面向数据的 Spring 应用程序轻松配置和访问 ApacheGeode。

  • Spring Data for Apache Solr - 为面向搜索的 Spring 应用程序轻松配置和访问 ApacheSolr。

  • Spring Data for Pivotal GemFire - 易于配置和访问枢纽 GemFire 为您的高度一致性,低延迟/高吞吐量,面向数据的 Spring 应用程序。

  • ……

1.3、Features(特性)

  1. 强大的repository仓储和自定义对象映射ORM抽象

  2. 从repository方法名称派生动态查询接口

  3. 实现Domain域基类提供基本属性

  4. 支持透明审计日志(创建,最后更改)

  5. 可以自定义repository代码

  6. 通过JavaConfig和自定义XML命名空间轻松实现 Spring集成

  7. 与Spring MVC控制器的高级集成

  8. 跨库持久性的实验支持

 

  • 模板制作

    • mongoTemplate、redisTemplate、jdbcTemplate……

模板提供存贮特定操作,例如保存、更新和删除单个记录或用于执行查询或映射/减少作业。

Spring Data JPA不提供模板,因为JPA实现本身已经是 JDBC API 之上的抽象层。JPA 的 EntityManager 是模板的对应物。异物转换由存储库实现处理。

  • 对象/数据存储映射

可以通过xml或者注解进行对象关系映射

一、JPA

 @Entity
 @Table(name = "TUSR")
 public class User {
     
     @Id
     private String id;
     
     @Column(name = "fn")
     private String name;
     
     private Date gmt_create;
     
     @OneToMany
     private List<Role> roles;
     ...
 }

二、MongoDB

 @Document(collection="usr")
 public class User {
 ​
     @Id
     private String id;
 ​
     @Field("fn")
     private String name;
 ​
     private Date gmt_create;
 ​
     private List<Role> roles;
     ...
 }

三、Neo4J

 @NodeEntity
 public class User {
 ​
     @Graphld
     private String id;
 ​
     private String name;
 ​
     @RelatedTo(type="has",direction=Direction.OUTGOING)
     private List<Role> roles;
     ...
 }

1.4、Repository 支持

Repository 提供了最基本的数据访问功能,其几个子接口则扩展了一些功能。他们的继承关系如下:

Repository :仅仅是一个标识,表名任何继承它的均为仓库接口类。

  • CrudRepository:继承 Repository,实现了一组 CRUD 相关的方法

  • PagingAndSortingRepository:继承 CrudRepository,实现了一组分页排序相关的方法

  • JpaRepository:继承 PagingAndSortingRepository,实现一组 API规范相关的方法

  • ……

自定义的 Repository 只需继承 XxxxRepository,这样自定义的 Repository 接口就具备了通用的数据访问控制层的能力。

2、Spring Data JPA

2.1、什么是JPA?

JPA 即Java Persistence API。

JPA 是一个基于O/R映射的标准规范(目前最新版本是JPA 2.2 )。所谓规范即只定义标准规则(如注解、接口),不提供实现,软件提供商可以按照标准规范来实现,而使用者只需按照规范中定义的方式来使用,而不用和软件提供商的实现打交道。

JPA的出现有两个原因:

简化现有Java EE和Java SE应用的对象持久化的开发工作;

Sun希望整合对ORM技术,实现持久化领域的统一。

JPA 的主要实现有Hibernate、EclipseLink 和OpenJPA 等,这也意味着我们只要使用JPA 来开发,无论是哪一个开发方式都是一样的。

操作步骤:

 jpa操作的操作步骤
     1.加载配置文件创建实体管理器工厂
       Persisitence:静态方法(根据持久化单元名称创建实体管理器工厂)
         createEntityMnagerFactory(持久化单元名称)
       作用:创建实体管理器工厂
       
     2.根据实体管理器工厂,创建实体管理器
       EntityManagerFactory :获取EntityManager对象
       方法:createEntityManager
       * 内部维护的很多的内容
         内部维护了数据库信息,
         维护了缓存信息
         维护了所有的实体管理器对象
         再创建EntityManagerFactory的过程中会根据配置创建数据库表
       * EntityManagerFactory的创建过程比较浪费资源
       特点:线程安全的对象
         多个线程访问同一个EntityManagerFactory不会有线程安全问题
       * 如何解决EntityManagerFactory的创建过程浪费资源(耗时)的问题?
       思路:创建一个公共的EntityManagerFactory的对象
       * 静态代码块的形式创建EntityManagerFactory
       
     3.创建事务对象,开启事务
       EntityManager对象:实体类管理器
         getTransaction : 创建事务对象
         presist : 保存
         merge  : 更新
         remove : 删除
         find/getReference : 根据id查询
         
       Transaction 对象 : 事务
         begin:开启事务
         commit:提交事务
         rollback:回滚
     4.增删改查操作
     5.提交事务
     6.释放资源
   
   i.搭建环境的过程
     1.创建maven工程导入坐标
     2.需要配置jpa的核心配置文件
       *位置:配置到类路径下的一个叫做 META-INF 的文件夹下
       *命名:persistence.xml
     3.编写客户的实体类
     4.配置实体类和表,类中属性和表中字段的映射关系
     5.保存客户到数据库中
   ii.完成基本CRUD案例
     persist : 保存
     merge : 更新
     remove : 删除
     find/getRefrence : 根据id查询
     
   iii.jpql查询
     sql:查询的是表和表中的字段
     jpql:查询的是实体类和类中的属性
     * jpql和sql语句的语法相似
     
     1.查询全部
     2.分页查询
     3.统计查询
     4.条件查询
     5.排序

 

2.2、JPA与JDBC关系

相同处:

  1. 都跟数据库操作有关,JPA是 JDBC 的升华,升级版。

  2. JDBC 和 JPA都是一组规范接口

  3. 都是由SUN官方推出的

不同处:

  1. JDBC是由各个关系型数据库实现的,JPA是由ORM框架实现。

  2. JDBC 使用 SQL 语句和数据库通信。JPA用面向对象方式,通过ORM框架来生成SQL,进行操作。

  3. JPA 在 JDBC 之上的,JPA也要依赖 JDBC 才能操作数据库。

图示对比:

 

JDBC 是我们最熟悉的用力啊操作数据库的技术,但是随之而来带来了一些问题:

  1. 需要面向SQL语句来操作数据库,开发人员学习成本更高。

  2. 数据库的移植性不高,不同数据库的SQL语句无法通用。

  3. Java对象和数据库类型映射是个麻烦事。

 

2.3、JPA 规范

但是Sun公司在JDK1.5提出了JPA:

JPA 全称 Java Persistence API(2019年重命名为 Jakarta Persistence API),是Sun官方提出的一种ORM规范。O:Object、R:Relational、M:mapping

作用

1、简化持久化操作的开发工作:让开发者从繁琐的 JDBC 和 SQL 代码中解脱出来,直接面向对象持久化操作。

2、Sun希望持久化技术能够统一,实现天下归一:如果你是基于 JPA 进行持久化你可以随意切换数据库。

 

该规范为我们提供了:

  1. ORM映射元数据:JPA支持XML和注解两种元数据的形式,元数据描述对象和表之间的映射关系,框架据此将实体对象持久化到数据库表中。

如:@Entity、@Table、@Id、@Column等注解!

  1. JPA 的 API:用来操作实体对象,执行CRUD操作,框架在后台替我们完成所有的事情,开发者从繁琐的 JDBC 和 SQL代码中解脱出来。

如:entityManager.merge(T t);

  1. JPQL查询语言:通过面向对象而非面向数据库的查询语言查询数据,避免程序的SQL语句紧密耦合。

如:from Student s where s.name = ?

 

所以:JPA仅仅是一种规范,也就是说 JPA 仅仅定义了一些接口,而接口是需要实现才能工作的。

 

2.4、Hibernate与JPA:

所以底层需要魔种实现,而Hibernate就是实现了 JPA 接口的ORM框架。

也就是说:JPA是一套ORM规范,Hibernate实现了JPA规范!

 

2.5、Mybatis与Hibernate

Mybatis:小巧、方便、高效、简单、直接、半自动!

Hibernate:强大、方便、高效、复杂、绕弯子、全自动!

Hibernate的优点:

1、Hibernate是全自动,hibernate完全可以通过对象关系模型实现对数据库的操作,拥有完整的JavaBean对象与数据库的映射结构来自动生成sql。

2、功能强大,数据库无关性好,O/R映射能力强,需要写的代码很少,开发速度很快。

3、有更好的二级缓存机制,可以使用第三方缓存。

4、数据库移植性良好。

5、hibernate拥有完整的日志系统,hibernate日志系统非常健全,涉及广泛,包括sql记录、关系异常、优化警告、缓存提示、脏数据警告等

Hibernate的缺点:

1、学习门槛高,精通门槛更高,程序员如何设计O/R映射,在性能和对象模型之间如何取得平衡,以及怎样用好Hibernate方面需要的经验和能力都很强才行

2、hibernate的sql很多都是自动生成的,无法直接维护sql;虽然有hql查询,但功能还是不及sql强大,见到报表等变态需求时,hql查询要虚,也就是说hql查询是有局限的;hibernate虽然也支持原生sql查询,但开发模式上却与orm不同,需要转换思维,因此使用上有些不方便。总之写sql的灵活度上hibernate不及mybatis。

场景:首先hibernate在国外更加流行,在业务相对简单的系统进行使用,随着微服务的流行spring data jpa也流行起来了。

Mybatis的优点:

1、易于上手和掌握,提供了数据库查询的自动对象绑定功能,而且延续了很好的SQL使用经验,对于没有那么高的对象模型要求的项目来说,相当完美。

2、sql写在xml里,便于统一管理和优化, 解除sql与程序代码的耦合。

3、提供映射标签,支持对象与数据库的orm字段关系映射

4、 提供对象关系映射标签,支持对象关系组建维护

5、提供xml标签,支持编写动态sql。

6、速度相对于Hibernate的速度较快

Mybatis的缺点:

1、关联表多时,字段多的时候,sql工作量很大。

2、sql依赖于数据库,导致数据库移植性差。

3、由于xml里标签id必须唯一,导致DAO中方法不支持方法重载。

4、对象关系映射标签和字段映射标签仅仅是对映射关系的描述,具体实现仍然依赖于sql。

5、DAO层过于简单,对象组装的工作量较大。

6、不支持级联更新、级联删除。

7、Mybatis的日志除了基本记录功能外,其它功能薄弱很多。

8、编写动态sql时,不方便调试,尤其逻辑复杂时。

9、提供的写动态sql的xml标签功能简单,编写动态sql仍然受限,且可读性低。

场景:在国内更加流行,在业务比较复杂的情况下进行使用。小巧灵活!

 

2.6、JPA注解

注解 解释
@Entity 声明类为实体或表。
@Table 声明表名。
@Basic 指定非约束明确的各个字段。
@Embedded 指定类或它的值是一个可嵌入的类的实例的实体的属性。
@Id 指定的类的属性,用于识别(一个表中的主键)。
@GeneratedValue 指定如何标识属性可以被初始化,例如自动、手动、或从序列表中获得的值。
@Transient 指定的属性,它是不持久的,即:该值永远不会存储在数据库中。
@Column 指定持久属性栏属性。
@SequenceGenerator 指定在@GeneratedValue注解中指定的属性的值。它创建了一个序列。
@TableGenerator 指定在@GeneratedValue批注指定属性的值发生器。它创造了的值生成的表。
@AccessType 这种类型的注释用于设置访问类型。如果设置@AccessType(FIELD),则可以直接访问变量并且不需要getter和setter,但必须为public。如果设置@AccessType(PROPERTY),通过getter和setter方法访问Entity的变量。
@JoinColumn 指定一个实体组织或实体的集合。这是用在多对一和一对多关联。
@UniqueConstraint 指定的字段和用于主要或辅助表的唯一约束。
@ColumnResult 参考使用select子句的SQL查询中的列名。
@ManyToMany 定义了连接表之间的多对多一对多的关系。
@ManyToOne 定义了连接表之间的多对一的关系。
@OneToMany 定义了连接表之间存在一个一对多的关系。
@OneToOne 定义了连接表之间有一个一对一的关系。
@NamedQueries 指定命名查询的列表。
@NamedQuery 指定使用静态名称的查询。

 

3、Hibernate快速搭建JPA

Hibernate示例

我们来实现一个Hibernate来实例感受一下:

01-springdata-jpa-hibernate

1、pom.xml

 <dependencies>
     <!--junit-->
     <dependency>
         <groupId>junit</groupId>
         <artifactId>junit</artifactId>
         <version>4.12</version>
     </dependency>
     <!--hibernate对jpa的支持包!-->
     <dependency>
         <groupId>org.hibernate</groupId>
         <artifactId>hibernate-entitymanager</artifactId>
         <version>5.6.0.Final</version>
         <type>pom</type>
     </dependency>
     <!--mysql and MariaDB-->
     <dependency>
         <groupId>mysql</groupId>
         <artifactId>mysql-connector-java</artifactId>
         <version>8.0.27</version>
     </dependency>
     <!-- log日志 -->
     <dependency>
         <groupId>log4j</groupId>
         <artifactId>log4j</artifactId>
         <version>1.2.17</version>
     </dependency>
 </dependencies>

对于这里有一个注意点,那就是我们的hibernate-entitymanager如果你使用了6.0.0.Alpha7或者更高的版本,可能会有很多改变,所以我推荐使用5.6.0.Final!

2、创建实体类!

 package com.yykk.pojo;
 ​
 import javax.persistence.*;
 ​
 /**
  * @author yykk
  */
 @Entity // 作为 hibernate实体类
 @Table(name = "tb_Customer") // 配置数据库表的名称,实体类中属性和表中字段的映射关系!
 public class Customer {
 ​
     /**
      * @Id:声明主键的配置
      * @GeneratedValue:配置主键的生成策略 strategy
      * GenerationType.IDENTITY :自增,mysql
      * * 底层数据库必须支持自动增长(底层数据库支持的自动增长方式,对id自增)
      * GenerationType.SEQUENCE : 序列,oracle
      * * 底层数据库必须支持序列
      * GenerationType.TABLE : jpa提供的一种机制,通过一张数据库表的形式帮助我们完成主键自增
      * GenerationType.AUTO : 由程序自动的帮助我们选择主键生成策略
      * @Column:配置属性和字段的映射关系 name:数据库表中字段的名称
      */
     @Id
     @GeneratedValue(strategy = GenerationType.IDENTITY)
     private Long id;
 ​
     @Column(name = "cust_name")
     private String custName;//客户名称
 ​
     @Column(name = "cust_source")
     private String custSource;//客户来源
 ​
     @Column(name = "cust_level")
     private String custLevel;//客户级别
 ​
     @Column(name = "cust_industry")
     private String custIndustry;//客户所属行业
 ​
     @Column(name = "cust_phone")
     private String custPhone;//客户的联系方式
 ​
     @Column(name = "cust_address")
     private String custAddress;//客户地址
 ​
 ​
     public Long getId() {
         return id;
     }
 ​
     public void setId(Long id) {
         this.id = id;
     }
 ​
     public String getCustName() {
         return custName;
     }
 ​
     public void setCustName(String custName) {
         this.custName = custName;
     }
 ​
     public String getCustSource() {
         return custSource;
     }
 ​
     public void setCustSource(String custSource) {
         this.custSource = custSource;
     }
 ​
     public String getCustLevel() {
         return custLevel;
     }
 ​
     public void setCustLevel(String custLevel) {
         this.custLevel = custLevel;
     }
 ​
     public String getCustIndustry() {
         return custIndustry;
     }
 ​
     public void setCustIndustry(String custIndustry) {
         this.custIndustry = custIndustry;
     }
 ​
     public String getCustPhone() {
         return custPhone;
     }
 ​
     public void setCustPhone(String custPhone) {
         this.custPhone = custPhone;
     }
 ​
     public String getCustAddress() {
         return custAddress;
     }
 ​
     public void setCustAddress(String custAddress) {
         this.custAddress = custAddress;
     }
 ​
     @Override
     public String toString() {
         return "Customer{" +
                 "id=" + id +
                 ", custName='" + custName + '\'' +
                 ", custSource='" + custSource + '\'' +
                 ", custLevel='" + custLevel + '\'' +
                 ", custIndustry='" + custIndustry + '\'' +
                 ", custPhone='" + custPhone + '\'' +
                 ", custAddress='" + custAddress + '\'' +
                 '}';
     }
 }

3、配置配置文件!

hibernate.xml

<?xml version='1.0' encoding='utf-8'?>
<!DOCTYPE hibernate-configuration PUBLIC
        "-//Hibernate/Hibernate Configuration DTD//EN"
        "http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
    <session-factory>
        <!--配置数据库的连接信息!-->
        <property name="connection.driver_class" >com.mysql.jdbc.Driver</property>
        <property name="connection.url">jdbc:mysql://localhost:3306/spring_data?characterEncoding=UTF-8;useSSL=false;useUnicode=true;serverTImezone=UTC</property>
        <property name="hibernate.connection.username">root</property>
        <property name="hibernate.connection.password">123456</property>
        <!-- 指定连接数据库的编码 -->
        <property name="connection.characterEncoding">utf8</property>
        <!-- 指定数据库方言 -->
        <property name="hibernate.dialect">org.hibernate.dialect.MySQLDialect</property>
        <!-- 显示Hibernate持久化操作所生成的SQL,默认为false -->
        <property name="show_sql">true</property>
        <!-- 将SQL脚本进行格式化后再输出,默认为false -->
        <property name="format_sql">true</property>
        <!-- 指定自动生成数据表的策略
                 默认为none 不自动生成
                 update  如果没有表会自动创建,有会自动检查更新
                 create  不管有没有这个表,每一次都会创建一次!
        -->
        <property name="hbm2ddl.auto">update</property>
        <!--配置方言:数据库连接类型!-->
        <property name="dialect">org.hibernate.dialect.MySQLDialect</property>
        <!-- DB schema will be updated if needed -->
        <!-- 指定pojo,进行表的ORM映射 -->
        <mapping class="com.yykk.pojo.Customer"/>
    </session-factory>

</hibernate-configuration>

对于我们的数据库选型,这里可以自由指定,打开Dialog方言类,按 ctrl + H 即可查看所有的选型!

如下所示:

4、测试类测试!

package com.Test;

import com.yykk.pojo.Customer;
import org.hibernate.HibernateException;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.Transaction;
import org.hibernate.boot.MetadataSources;
import org.hibernate.boot.registry.StandardServiceRegistry;
import org.hibernate.boot.registry.StandardServiceRegistryBuilder;
import org.junit.Before;
import org.junit.Test;

import java.util.List;

public class HibernateTest {

    // session 工厂 Session:数据库会话,代码持久化操作数据库的一种桥梁!
    private SessionFactory sessionFactory;

    @Before
    public void init() {
        StandardServiceRegistry registry = new StandardServiceRegistryBuilder().configure("hibernate.xml").build();

        // 2、根据服务注册类创建一个元数据资源集,同时构建元数据并生成应用一般唯一的session工厂
        sessionFactory = new MetadataSources(registry).buildMetadata().buildSessionFactory();
    }

    /**
     * 新增数据
     */
    @Test
    public void testInsert() {
        // 进行持久化操作!
        try {
            Session session = sessionFactory.openSession();
            Transaction transaction = session.beginTransaction();

            Customer customer = new Customer();
            customer.setCustName("yykk");
            customer.setCustPhone("123456");
            session.save(customer);
            transaction.commit();
        } catch (HibernateException e) {
            e.printStackTrace();
        } finally {
            sessionFactory.close();
        }
    }

    /**
     * 查询数据
     */
    @Test
    public void testQuery() {
        // 进行持久化操作!
        try {
            Session session = sessionFactory.openSession();
            Transaction transaction = session.beginTransaction();

            Customer customer = session.find(Customer.class, 1L);
            System.out.println(customer);
            transaction.commit();
        } catch (HibernateException e) {
            e.printStackTrace();
        } finally {
            sessionFactory.close();
        }
    }

    /**
     * lazy 查询数据
     */
    @Test
    public void testQuery_lazy() {
        // 进行持久化操作!
        try {
            Session session = sessionFactory.openSession();
            Transaction transaction = session.beginTransaction();

            Customer customer = session.load(Customer.class, 1L);
            System.out.println(customer);
            transaction.commit();
        } catch (HibernateException e) {
            e.printStackTrace();
        } finally {
            sessionFactory.close();
        }
    }

    /**
     * 更新数据
     */
    @Test
    public void testUpdate() {
        // 进行持久化操作!
        try {
            Session session = sessionFactory.openSession();
            Transaction transaction = session.beginTransaction();

            Customer customer = new Customer();
            customer.setCustLevel("level1");
            customer.setId(1L);
            session.saveOrUpdate(customer);  // 如果存在id就插入,不存在就更新!  或者使用save 、update!
            transaction.commit();
        } catch (HibernateException e) {
            e.printStackTrace();
        } finally {
            sessionFactory.close();
        }
    }

    /**
     * 删除数据
     */
    @Test
    public void testRemove() {
        // 进行持久化操作!
        try {
            Session session = sessionFactory.openSession();
            Transaction transaction = session.beginTransaction();

            Customer customer = new Customer();
            customer.setId(1L);
            session.remove(customer);
            transaction.commit();
        } catch (HibernateException e) {
            e.printStackTrace();
        } finally {
            sessionFactory.close();
        }
    }

    /**
     * HQL查询!
     */
    @Test
    public void testHQL() {
        // 进行持久化操作!
        try {
            Session session = sessionFactory.openSession();
            Transaction transaction = session.beginTransaction();

            //String hql = "from Customer";
            String hql = "from Customer where id = :id";  // 或者可以使用?进行传参!
            List<Customer> resultList = session.createQuery(hql, Customer.class)
                    .setParameter("id",1L)
                    .getResultList();
            System.out.println(resultList);
            transaction.commit();
        } catch (HibernateException e) {
            e.printStackTrace();
        } finally {
            sessionFactory.close();
        }
    }
}

Jpa实例

1、pom依赖、实体类都如上所示!

2、配置文件

前提配置文件需要放在META-INF目录下:(如果是默认没有,就在resources目录下创建!resources/META-INF/persistence.xml)

META-INF/persistence.xml

<!--?xml version="1.0" encoding="UTF-8"?-->
<persistence xmlns="http://java.sun.com/xml/ns/persistence" version="2.0">
    <!--需要配置persistence-unit节点
        持久化单元:
            name:持久化单元名称
            transaction-type:事务管理的方式
                    JTA:分布式事务管理
                    RESOURCE_LOCAL:本地事务管理
    -->
    <persistence-unit name="hibernateJpa" transaction-type="RESOURCE_LOCAL">
        <!--jpa的实现方式 -->
        <provider>org.hibernate.jpa.HibernatePersistenceProvider</provider>
        <!-- 需要进行ORM映射的pojo类 -->
        <class>com.yykk.pojo.Customer</class>
        <!--可选配置:配置jpa实现方的配置信息-->
        <properties>
            <!-- 数据库信息
                用户名,javax.persistence.jdbc.user
                密码,  javax.persistence.jdbc.password
                驱动,  javax.persistence.jdbc.driver
                数据库地址   javax.persistence.jdbc.url
            -->
            <property name="javax.persistence.jdbc.user" value="root"/>
            <property name="javax.persistence.jdbc.password" value="123456"/>
            <property name="javax.persistence.jdbc.driver" value="com.mysql.jdbc.Driver"/>
            <property name="javax.persistence.jdbc.url" value="jdbc:mysql:///spring_data"/>

            <!--配置jpa实现方(hibernate)的配置信息
                显示sql           :   false|true
                自动创建数据库表    :  hibernate.hbm2ddl.auto
                        create      : 程序运行时创建数据库表(如果有表,先删除表再创建)
                        update      :程序运行时创建表(如果有表,不会创建表)
                        none        :不会创建表

            -->
            <property name="hibernate.show_sql" value="true"/>
            <property name="hibernate.hbm2ddl.auto" value="update"/>
            <property name="dialect" value="org.hibernate.dialect.MySQLDialect"/>
        </properties>
    </persistence-unit>

    <!-- openjpa实现,在这里使用需要引入依赖! -->
    <persistence-unit name="openJpa" transaction-type="RESOURCE_LOCAL">
        <!--jpa的实现方式 -->
        <provider>org.apache.openjpa.persistence.PersistenceProviderImpl</provider>
        <!-- 需要进行ORM映射的pojo类 -->
        <class>com.yykk.pojo.Customer</class>
        <!--可选配置:配置jpa实现方的配置信息-->
        <properties>
            <!-- 数据库信息
                用户名,javax.persistence.jdbc.user
                密码,  javax.persistence.jdbc.password
                驱动,  javax.persistence.jdbc.driver
                数据库地址   javax.persistence.jdbc.url
            -->
            <property name="javax.persistence.jdbc.user" value="root"/>
            <property name="javax.persistence.jdbc.password" value="123456"/>
            <property name="javax.persistence.jdbc.driver" value="com.mysql.jdbc.Driver"/>
            <property name="javax.persistence.jdbc.url" value="jdbc:mysql:///spring_data"/>

            <!-- 配置jpa实现方(openjpa)的配置信息 -->
            <property name="openjpa.jdbc.SynchronizeMappings" value="buildSchema(ForeignKeys=true)"/>
        </properties>
    </persistence-unit>
</persistence>

openJpa依赖

<!--openjpa 实现-->
<dependency>
    <groupId>org.apache.openjpa</groupId>
    <artifactId>openjpa-all</artifactId>
    <version>3.2.1</version>
</dependency>

3、测试!

package com.Test;

import com.yykk.pojo.Customer;
import org.junit.Before;
import org.junit.Test;

import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import javax.persistence.EntityTransaction;
import javax.persistence.Persistence;

public class JpaTest {

    EntityManagerFactory factory;

    @Before
    public void before () {
        factory = Persistence.createEntityManagerFactory("hibernateJpa");
    }

    /**
     * 插入数据
     */
    @Test
    public void testInsert() {
        EntityManager entityManager = factory.createEntityManager();
        EntityTransaction transaction = entityManager.getTransaction();
        transaction.begin();

        Customer customer = new Customer();
        customer.setCustName("张三");

        entityManager.persist(customer);
        transaction.commit();
    }

    /**
     * 查询数据
     */
    @Test
    public void testQuery() {
        EntityManager entityManager = factory.createEntityManager();
        EntityTransaction transaction = entityManager.getTransaction();
        transaction.begin();

        Customer customer = entityManager.find(Customer.class, 3L);
        System.out.println(customer);
        transaction.commit();
    }

    /**
     * lazy查询
     */
    @Test
    public void testQuery_lazy() {
        EntityManager entityManager = factory.createEntityManager();
        EntityTransaction transaction = entityManager.getTransaction();
        transaction.begin();

        Customer customer = entityManager.getReference(Customer.class, 3L);
        System.out.println(customer);
        transaction.commit();
    }

    /**
     * 更新数据/插入
     */
    @Test
    public void testUpdate() {
        EntityManager entityManager = factory.createEntityManager();
        EntityTransaction transaction = entityManager.getTransaction();
        transaction.begin();

        Customer customer = new Customer();
        customer.setId(4L);
        customer.setCustName("李四");
        /**
         * merge()
         * 如果指定了主键:
         *      更新:先查询,看是否有变化!如果有更新,没有就不更新!
         * 如果没有指定主键:
         *      插入:或者是指定了不存在的主键!
         */
        entityManager.merge(customer);
        transaction.commit();
    }

    /**
     * JPQL更新数据
     */
    @Test
    public void testUpdate_JPQL() {
        EntityManager entityManager = factory.createEntityManager();
        EntityTransaction transaction = entityManager.getTransaction();
        transaction.begin();

        String jpql = "update Customer set custName=:name where id=:id";
        entityManager.createQuery(jpql)
                .setParameter("name","张三")
                .setParameter("id",4L)
                .executeUpdate();

        /**
         * 当然了,如果你的业务逻辑十分复杂,只能使用sql进行查询!示例如下:
         * update 数据库表名
         * createNativeQuery :创建一个本地查询!
          */
        //String sql = "update tb_Customer set custName=:name where id=:id";
        //entityManager.createNativeQuery(sql)
        //        .setParameter("name","张三")
        //        .setParameter("id",4L)
        //        .executeUpdate();

        transaction.commit();
    }

    /**
     * 删除数据
     */
    @Test
    public void testRemove() {
        EntityManager entityManager = factory.createEntityManager();
        EntityTransaction transaction = entityManager.getTransaction();
        transaction.begin();

        Customer customer = entityManager.find(Customer.class,6L);

        entityManager.remove(customer); // 如果你直接进行删除,会报错:Removing a detached instance ,不可以删除已经持久化的数据!
        transaction.commit();
    }
}

4、JPA四大状态

 

JPA的四种状态

  • 临时状态:刚创建出来,没有与entityManager发生关系,没有被持久化,不处于entityManager中的对象。

  • 持久状态:与entityManager发生关系,已经被持久化,您可以把持久化状态当做实实在在的数据库记录。

  • 删除状态:执行remove方法,事务提交之前。

  • 游离状态:游离状态就是提交到数据库后,事务commit后实体的状态,应为事务已经提交了,此时实体的属性任你如何改变,也不会同步到数据库,因为游离是没人管的孩子,不在持久化上下文中。

使用代码进行演示,帮助理解:

/**
 * 游离状态
 */
@Test
public void testStatus() {
    EntityManager entityManager = factory.createEntityManager();
    EntityTransaction transaction = entityManager.getTransaction();
    transaction.begin();
    Customer customer = new Customer(); // 临时状态(瞬时状态)
    customer.setId(5l);  // 游离状态

    customer = entityManager.find(Customer.class,6L);  // 持久状态

    entityManager.remove(customer); // 删除状态(销毁状态)
    transaction.commit();
}

如图所示:

  • public void persist(Object entity)

    • persist方法可以将实例转换为managed(托管)状态。在调用flush()方法或者提交事务之后,实例将会被插入到数据库中!

对不同状态下的实例A,persist会产生以下操作:

  1. 如果A是一个new状态的实体,它会转为managed状态

  2. 如果A是一个managed状态的实体,它的状态不会发生任何改变。但是系统仍会在数据库执行INSERT操作!

  3. 如果A是一个removed(删除)状态的实体,他将会转换为受控状态!(就是不会再去执行了!)

  4. 如果A是一个detached(分离)状态的实体,该方法会抛出IllegalArgumentException异常,具体异常根据不同的 JPA 实现有关!

  • public void merge(Object entity)

    • merge方法的主要作用是将用户对一个detached状态实体的修改进行归档,归档后产生一个新的managed状态对象。

对不同状态下的实例A,merge会产生一下操作:

  1. 如果A是个detached状态的实体, 该方法会将A的修改提交到数据库,并返回一个新的managed状态的实例A2:

  2. 如果A是个new状态的实体, 该方法会产生个根据A广 生的managed状态实体A2;

  3. 如果A是个managed状态的实体, 它的状态不会发生任何改变。但是系统仍会在数据库执行UPDATE操作:

  4. 如果A是个removed状态的实体, 该方法会 抛出IllegalArgumentException异常。

  • public void refresh(Object entity)

    • refresh方法可以保证当前的实例与数据库中的实例的内容致。

对 不同状态下的实例A. refresh会产生以下操作:

  1. 如果A是- 个new状态的实例,不会发生任何操作, 但有可能会抛出异常,具体情况根据不同JPA实现有关!

  2. 如果A是一个managed状态的实例,它的属性将会和数据库中的数据同步。

  3. 如果A是个removed状态的实例, 该方法将会抛出异常Entity not managed

  4. 如果A是一个detached状态的实体,该方法会抛出异常。

  • public void remove(Object entity)

    • remove方法可以将实体转换为removed状态,并且在调用flush()方法或提交事物后删除数据库中的数据。

对不同状态下的实例A. remove会产生以下操作:

  1. 如果A是一个new状态的实例,A的状态不会发生任何改变,但系统仍会在数据库中执行DELETE语句;

  2. 如果A是个managed状态的实例, 它的状态 会转换为removed;

  3. 如果A是个removed状 态的实例,不会发生任何操作:

  4. 如果A是个detached状态的实体, 该方法将会抛出异常。

缓存测试

/**
 * 缓存测试
 */
@Test
public void testCache() {
    EntityManager entityManager = factory.createEntityManager();
    EntityTransaction transaction = entityManager.getTransaction();
    transaction.begin();

  	// 这里的缓存是一级缓存,只查询了一次!
    Customer customer = entityManager.find(Customer.class,3L);
    Customer customer2 = entityManager.find(Customer.class,3L);

    transaction.commit();
}

 

5、Spring data Jpa

5.1、什么是spring data jpa?

官网:https://docs.spring.io/spring-data/jpa/docs/current/reference/html/

Spring Data JPA, part of the larger Spring Data family, makes it easy to easily implement JPA based repositories. This module deals with enhanced support for JPA based data access layers. It makes it easier to build Spring-powered applications that use data access technologies.

Implementing a data access layer of an application has been cumbersome for quite a while. Too much boilerplate code has to be written to execute simple queries as well as perform pagination, and auditing. Spring Data JPA aims to significantly improve the implementation of data access layers by reducing the effort to the amount that’s actually needed. As a developer you write your repository interfaces, including custom finder methods, and Spring will provide the implementation automatically.

 

SpringDataJPA 是更大的 SpringData 系列的一部分,它使得基于 JPA 的存储库的实现更加容易。该模块处理对基于 JPA 的数据访问层(repositories)的增强支持。它使得构建使用数据访问技术的 Spring 驱动的应用程序变得更加容易。

实现应用程序的数据访问层已经很麻烦了。为了执行简单的查询以及执行分页和审计,必须编写太多的样板代码。Spring Data JPA 旨在通过将工作量减少到实际需要的数量来显著改进数据访问层的实现。作为开发人员,您编写存储库接口,包括自定义查找器(finder)方法,Spring 将自动提供实现。

简单理解:

spring data jpa是spring提供的一套简化 JPA开发的框架 ,按照约定好的规则进行【方法命名】去写dao层接口,就可以在不写接口的情况下,实现对数据库的访问和操作。同时提供了很多除了CRUD之外的功能,如分页、排序、复杂查询等等!

 

Spring data Jpa 让我们解脱了DAO层的操作,基本上所有CRUD都可以依赖于它来实现,在实际的工作工程中,推荐使用Spring Data JPA + ORM (如:hibernate)完成操作,这样在切换不同的ORM框架时提供了极大的方便,同时也使数据库层操作更加简单,方便解耦!

 

5.2、特性(Features)

  • Sophisticated support to build repositories based on Spring and JPA

  • Support for Querydsl predicates and thus type-safe JPA queries

  • Transparent auditing of domain class

  • Pagination support, dynamic query execution, ability to integrate custom data access code

  • Validation of @Query annotated queries at bootstrap time

  • Support for XML based entity mapping

  • JavaConfig based repository configuration by introducing @EnableJpaRepositories.

 

SpringData JPA 极大简化了数据库访问层代码。如何简化呢?使用springdataJpa,我们的dao层只需要写接口,就自动有了增删查改、分页查询等方法。

这里Java application通过dao层进行访问,其实就是拿到了Jdk的动态代理!

 

5.3、Spring Data JPA实例

我们来实现一个基于Spring Data JPA的示例感受一下和之前单独使用的区别:

pom.xml依赖

1、最好在父maven项目中设置spring data统一版本管理依赖:因为不同的spring data子项目发布时间版本不一致,你自己维护很麻烦,这样不同的spring data子项目能保证是统一版本!

定义统一的springdata子项目的版本,在总的pom.xml进行添加!

    <!-- 统一管理SpringData子项目的版本!-->
    <dependencyManagement>
        <dependencies>
            <dependency>
                <groupId>org.springframework.data</groupId>
                <artifactId>spring-data-bom</artifactId>
                <version>2021.1.3</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>
        </dependencies>
    </dependencyManagement>

2、然后在我们的子项目中进行导入使用!

pom.xml

<dependencies>
    <dependency>
        <groupId>org.springframework.data</groupId>
        <artifactId>spring-data-jpa</artifactId>
    </dependency>
    <!--junit-->
    <dependency>
        <groupId>junit</groupId>
        <artifactId>junit</artifactId>
        <version>4.12</version>
    </dependency>
    <!--hibernate对jpa的支持包!-->
    <dependency>
        <groupId>org.hibernate</groupId>
        <artifactId>hibernate-entitymanager</artifactId>
        <version>6.0.0.Alpha6</version>
    </dependency>
    <!--mysql and MariaDB-->
    <dependency>
        <groupId>mysql</groupId>
        <artifactId>mysql-connector-java</artifactId>
        <version>8.0.27</version>
    </dependency>
    <!--连接池-->
    <dependency>
        <groupId>com.alibaba</groupId>
        <artifactId>druid</artifactId>
        <version>1.2.8</version>
    </dependency>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-test</artifactId>
        <version>5.3.15</version>
    </dependency>
</dependencies>

在这里因为我们是创建的maven工程,所以有必要添加hibernate支持!如果是springboot项目的话,可以不去添加依赖!

02-springdata-jpa

3、创建实体类!

package com.yykk.pojo;

import javax.persistence.*;

/**
 * @author yykk
 */
@Entity // 作为 hibernate实体类
@Table(name = "tb_Customer") // 配置数据库表的名称,实体类中属性和表中字段的映射关系!
public class Customer {

    /**
     * @Id:声明主键的配置
     * @GeneratedValue:配置主键的生成策略 strategy
     * GenerationType.IDENTITY :自增,mysql
     * * 底层数据库必须支持自动增长(底层数据库支持的自动增长方式,对id自增)
     * GenerationType.SEQUENCE : 序列,oracle
     * * 底层数据库必须支持序列
     * GenerationType.TABLE : jpa提供的一种机制,通过一张数据库表的形式帮助我们完成主键自增
     * GenerationType.AUTO : 由程序自动的帮助我们选择主键生成策略
     * @Column:配置属性和字段的映射关系 name:数据库表中字段的名称
     */
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;

    @Column(name = "cust_name")
    private String custName;//客户名称

    @Column(name = "cust_source")
    private String custSource;//客户来源

    @Column(name = "cust_level")
    private String custLevel;//客户级别

    @Column(name = "cust_industry")
    private String custIndustry;//客户所属行业

    @Column(name = "cust_phone")
    private String custPhone;//客户的联系方式

    @Column(name = "cust_address")
    private String custAddress;//客户地址


    public Long getId() {
        return id;
    }

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

    public String getCustName() {
        return custName;
    }

    public void setCustName(String custName) {
        this.custName = custName;
    }

    public String getCustSource() {
        return custSource;
    }

    public void setCustSource(String custSource) {
        this.custSource = custSource;
    }

    public String getCustLevel() {
        return custLevel;
    }

    public void setCustLevel(String custLevel) {
        this.custLevel = custLevel;
    }

    public String getCustIndustry() {
        return custIndustry;
    }

    public void setCustIndustry(String custIndustry) {
        this.custIndustry = custIndustry;
    }

    public String getCustPhone() {
        return custPhone;
    }

    public void setCustPhone(String custPhone) {
        this.custPhone = custPhone;
    }

    public String getCustAddress() {
        return custAddress;
    }

    public void setCustAddress(String custAddress) {
        this.custAddress = custAddress;
    }

    @Override
    public String toString() {
        return "Customer{" +
                "id=" + id +
                ", custName='" + custName + '\'' +
                ", custSource='" + custSource + '\'' +
                ", custLevel='" + custLevel + '\'' +
                ", custIndustry='" + custIndustry + '\'' +
                ", custPhone='" + custPhone + '\'' +
                ", custAddress='" + custAddress + '\'' +
                '}';
    }
}

4、配置xml,resources/springdata.xml,如下配置:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:jpa="http://www.springframework.org/schema/data/jpa"
       xmlns:tx="http://www.springframework.org/schema/tx"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       https://www.springframework.org/schema/beans/spring-beans.xsd
       http://www.springframework.org/schema/data/jpa
       https://www.springframework.org/schema/data/jpa/spring-jpa.xsd
       http://www.springframework.org/schema/tx
       https://www.springframework.org/schema/tx/spring-tx.xsd">

    <!--用户整合jpa @EnableJpaRepositories-->
    <jpa:repositories base-package="com.yykk.repositories"
                      entity-manager-factory-ref="entityManagerFactory"
                      transaction-manager-ref="transactionManager" />

    <!--EntityManagerFactory-->
    <bean name="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
        <property name="jpaVendorAdapter">
            <!--hibernate实现-->
            <bean class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
                <!--生成数据库表-->
                <property name="showSql" value="true"></property>
                <property name="generateDdl" value="true"></property>
            </bean>
        </property>
        <!--设置实体类的包-->
        <property name="packagesToScan" value="com.yykk.pojo"></property>
        <property name="dataSource" ref="dataSource"></property>
    </bean>

    <!--数据源-->
    <bean class="com.alibaba.druid.pool.DruidDataSource" name="dataSource">
        <property name="username" value="root"/>
        <property name="password" value="123456"/>
        <property name="url" value="jdbc:mysql://localhost:3306/spring_data?characterEncoding=UTF-8"/>  <!--这里要注意:在我配置了其他属性之后,就加载不出来了!-->
        <property name="driverClassName" value="com.mysql.jdbc.Driver"/>
    </bean>

    <!--声明式事务-->
    <bean class="org.springframework.orm.jpa.JpaTransactionManager" name="transactionManager">
        <property name="entityManagerFactory" ref="entityManagerFactory"></property>
    </bean>

    <!--启动注解方式的声明式事务-->
    <tx:annotation-driven transaction-manager="transactionManager"></tx:annotation-driven>

</beans>

5、创建repository/repositories 接口!

package com.yykk.repositories;

import com.yykk.pojo.Customer;
import org.springframework.data.repository.CrudRepository;

public interface CustomerRepositories extends CrudRepository<Customer,Long> {
}

6、测试!

import com.yykk.pojo.Customer;
import com.yykk.repositories.CustomerRepositories;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;

import java.util.Optional;

@ContextConfiguration("/springdata.xml")
@RunWith(SpringJUnit4ClassRunner.class)
public class SpringdataJpaTest {

    @Autowired
    CustomerRepositories repositories;

    @Test
    public void test_Query() {
        Optional<Customer> id = repositories.findById(3L);
        System.out.println(id.get());
    }

    @Test
    public void test_Insert() {
        Customer customer = new Customer();
        customer.setCustName("李四");
        customer.setCustAddress("南京");
        repositories.save(customer);
    }

    @Test
    public void test_Update() {
        Customer customer = new Customer();
        customer.setId(7L);
        customer.setCustName("王五");
        repositories.save(customer);
    }

    /**
     * 在这里delete方法,可以发现一个问题:
     *  1、底层会帮我们先进行查询一次,在进行删除!
     *  2、这样属性就不是游离状态,而是持久状态了!
     */
    @Test
    public void test_Delete() {
        Customer customer = new Customer();
        customer.setId(7L);
        customer.setCustName("王五");
        repositories.delete(customer);
    }
}

7、使用JavaConfig进行配置!

config/SpringDataJPAConfig

package com.yykk.config;

import com.alibaba.druid.pool.DruidDataSource;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.jpa.repository.config.EnableJpaRepositories;
import org.springframework.orm.jpa.JpaTransactionManager;
import org.springframework.orm.jpa.JpaVendorAdapter;
import org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean;
import org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter;
import org.springframework.transaction.annotation.EnableTransactionManagement;

import javax.persistence.EntityManagerFactory;
import javax.sql.DataSource;

@Configuration
@EnableJpaRepositories(basePackages = "com.yykk.repositories")
@EnableTransactionManagement
public class SpringDataJPAConfig {

    @Bean
    public DataSource dataSource() {
        DruidDataSource dataSource = new DruidDataSource();
        dataSource.setUsername("root");
        dataSource.setPassword("123456");
        dataSource.setDriverClassName("com.mysql.jdbc.Driver");
        dataSource.setUrl("jdbc:mysql://localhost:3306/spring_data?characterEncoding=UTF-8");
        return dataSource;
    }

    @Bean
    public JpaTransactionManager transactionManager(EntityManagerFactory emf) {
        return new JpaTransactionManager(emf);
    }

    @Bean
    public JpaVendorAdapter jpaVendorAdapter() {
        HibernateJpaVendorAdapter jpaVendorAdapter = new HibernateJpaVendorAdapter();
        //jpaVendorAdapter.setDatabase(Database.MYSQL);
        jpaVendorAdapter.setGenerateDdl(true);
        jpaVendorAdapter.setShowSql(true);
        return jpaVendorAdapter;
    }

    @Bean
    public LocalContainerEntityManagerFactoryBean entityManagerFactory() {
        LocalContainerEntityManagerFactoryBean lemfb = new LocalContainerEntityManagerFactoryBean();
        lemfb.setDataSource(dataSource());
        lemfb.setJpaVendorAdapter(jpaVendorAdapter());
        lemfb.setPackagesToScan("com.yykk.pojo");
        return lemfb;
    }
}

8、测试!

 

5.4、使用Spring Data Repositories

Spring Data Repositories 抽象的目标是显着减少各种持久化存储实现数据访问层所需的样板代码量!

CrudRepository

// 用力啊插入和修改 ,有主键就是修改,没有就是新增!
// 获得插入后自增id,获得返回值
<S extends T> S save(S entity);

// 通过集合保存多个实体
<S extends T> Iterable<S> saveAll(Iterable<S> entities);

// 通过主键查询实体
Optional<T> findById(ID id);

// 通过主键查询id,返回boolean
boolean existsById(ID id);

// 查询所有!
Iterable<T> findAll();

// 通过集合的主键、查询多个实体,返回集合!
Iterable<T> findAllById(Iterable<ID> ids

// 查询总数量
long count();

// 根据id进行删除
void deleteById(ID id);

// 根据实体进行删除
void delete(T entity);

// 删除多个
void deleteAllById(Iterable<? extends ID> ids);

// 删除多个,传入集合实体!
void deleteAll(Iterable<? extends T> entities);

// 删除全部!
void deleteAll();

 

5.5、实现分页与排序

1、首先我们需要更改的是我们的repository继承PagingAndSortingRepository,其他配置相同即可!

 package com.yykk.repositories;
 ​
 import com.yykk.pojo.Customer;
 import org.springframework.data.repository.PagingAndSortingRepository;
 ​
 public interface CustomerRepositories extends PagingAndSortingRepository<Customer,Long> {
 }

2、测试!

 import com.yykk.config.SpringDataJPAConfig;
 import com.yykk.pojo.Customer;
 import com.yykk.repositories.CustomerRepositories;
 import org.junit.Test;
 import org.junit.runner.RunWith;
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.data.domain.Page;
 import org.springframework.data.domain.PageRequest;
 import org.springframework.data.domain.Sort;
 import org.springframework.test.context.ContextConfiguration;
 import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
 ​
 @ContextConfiguration(classes = SpringDataJPAConfig.class)
 @RunWith(SpringJUnit4ClassRunner.class)
 public class SpringDataJPAPagingAndSortTest {
 ​
     @Autowired
     CustomerRepositories repositories;
 ​
     @Test
     public void testPaging() {
         Page<Customer> all = repositories.findAll(PageRequest.of(0, 2));
         System.out.println(all.getTotalElements());
         System.out.println(all.getTotalPages());
         System.out.println(all.getContent());
     }
 ​
     @Test
     public void testSort() {
         Sort sort = Sort.by("id").descending();
         Iterable<Customer> all = repositories.findAll(sort);
         System.out.println(all);
     }
 ​
     @Test
     public void testSortTypeSafe() {
         Sort.TypedSort<Customer> sortType = Sort.sort(Customer.class);
 ​
         Sort sort = sortType.by(Customer::getId).ascending()
                 .and(sortType.by(Customer::getCustName)).descending();
 ​
         Iterable<Customer> all = repositories.findAll(sort);
         System.out.println(all);
     }
 }

5.6、知识点总结

 底层数据库操作笼统讲就是增删改查,例如JDBC模板,Hibernate,Mybaties,JPA。接下来尝试springBootJPA旅程
 ​
 1. @Column(nullable = false)
 ​
  内容扩展
 ​
 unique=true是指这个字段的值在这张表里不能重复,所有记录值都要唯一,就像主键那样。
 ​
 nullable=false是这个字段在保存时必需有值,不能还是null值就调用save去保存入库。
 ​
 这两个用法是不同的,需要看个人需要,互相不可取代,根据个人需要可以两个都设置也可以只设置其中一个。
 ​
 hibernate常用方法介绍:
 ​
 delete(Object entity) 删除指定的持久化实例在程序中一般先用    Assert.notNull和 Assert.isTrue断言entity是否为空 和 entity的id是否大于0,否则事务回滚。再用get(Class entityClass,Serializable id,LockMode lockMode)加锁查询出持久化实例,一般用lockMode.update悲观锁,最后用delete(Object entity)来删除此实例。
 ​
 deleteAll(Collection entities) 删除集合内全部持久化实例entities必须为持久化实例,否则报数据库异常错误。
 ​
 find(String queryString) 根据HQL查询字符串来返回实例集合find方法在执行时会先查找缓存,如果缓存找不到再查找数据库,如果再找不到就会返回null。
 ​
 get(Class entityClass,Serializable id)根据主键加载特定持久化实例在程序中一般先用     Assert.isTrue断言id是否大于0,若大于0继续执行,若查到数据则返回实例,否则返回空不同于load,load若有数据则返回实例,否则报出ObjectNotFoundEcception异常,相比来说get效率高些。
 ​
 save(Object entity) 保存新的实例,在程序中一般先用    Assert.notNull断言实体是否为空,在进行保存。
 ​
 ​
 ​
 @Query
 ​
 1. 一个使用@Query注解的简单例子
 ​
 @Query(value = "select name,author,price from Book b where b.price>?1 and b.price findByPriceRange(long price1, long price2);
  
 ​
 2.  Like表达式
 ​
 @Query(value = "select name,author,price from Book b where b.name like %:name%")
 List findByNameMatch(@Param("name") String name);
  
 ​
 3. 使用Native SQL Query
 ​
 所谓本地查询,就是使用原生的sql语句(根据数据库的不同,在sql的语法或结构方面可能有所区别)进行查询数据库的操作。
 ​
 @Query(value = "select * from book b where b.name=?1", nativeQuery = true)
 List findByName(String name);
  
 ​
 4. 使用@Param注解注入参数
 ​
 @Query(value = "select name,author,price from Book b where b.name = :name AND b.author=:author AND b.price=:price")
 List findByNamedParam(@Param("name") String name, @Param("author") String author,
         @Param("price") long price);
  
 ​
 5. SPEL表达式(使用时请参考最后的补充说明)
 ​
    '#{#entityName}'值为'Book'对象对应的数据表名称(book)。
 ​
 public interface BookQueryRepositoryExample extends Repository{
        @Query(value = "select * from #{#entityName} b where b.name=?1", nativeQuery = true)
        List findByName(String name);
 ​
 }
  
 ​
 6. 一个较完整的例子
 ​
 public interface BookQueryRepositoryExample extends Repository {
     @Query(value = "select * from Book b where b.name=?1", nativeQuery = true) 
     List findByName(String name);// 此方法sql将会报错(java.lang.IllegalArgumentException),看出原因了吗,若没看出来,请看下一个例子
 ​
     @Query(value = "select name,author,price from Book b where b.price>?1 and b.price findByPriceRange(long price1, long price2);
 ​
     @Query(value = "select name,author,price from Book b where b.name like %:name%")
     List findByNameMatch(@Param("name") String name);
 ​
     @Query(value = "select name,author,price from Book b where b.name = :name AND b.author=:author AND b.price=:price")
     List findByNamedParam(@Param("name") String name, @Param("author") String author,
             @Param("price") long price);
 ​
 }
  
 ​
 7.  解释例6中错误的原因:
 ​
      因为指定了nativeQuery = true,即使用原生的sql语句查询。使用java对象'Book'作为表名来查自然是不对的。只需将Book替换为表名book。
 ​
 @Query(value = "select * from book b where b.name=?1", nativeQuery = true)
 List findByName(String name);
  
 ​
  
 ​
 补充说明(2017-01-12):
 ​
   有同学提出来了,例子5中用'#{#entityName}'为啥取不到值啊?
 ​
   先来说一说'#{#entityName}'到底是个啥。从字面来看,'#{#entityName}'不就是实体类的名称么,对,他就是。
 ​
   实体类Book,使用@Entity注解后,spring会将实体类Book纳入管理。默认'#{#entityName}'的值就是'Book'。
 ​
   但是如果使用了@Entity(name = "book")来注解实体类Book,此时'#{#entityName}'的值就变成了'book'。
 ​
   到此,事情就明了了,只需要在用@Entity来注解实体类时指定name为此实体类对应的表名。在原生sql语句中,就可以把'#{#entityName}'来作为数据表名使用。

 

6、自定义操作(JPQL / SQL)

在我们经过了上面的学习,我们会发现一个问题:那就是我们所支持的就是一些简单的增删查改等等的操作,对于复杂的一些操作并不支持,所以我们也需要进行一些自定义,可以通过SQL或者 JPQL进行自定义操作!

自定义操作:

1、JPQL(原生SQL)

  • @Query

    • 查询如果返回单个实体,就使用pojo类进行接收即可,如果是多个就使用list进行接收!

    • 参数设置方式

      • 索引:?数字

      • 具名::参数名 结合@Param注解指定参数名称 |

    • 增删改:

      • 要加上事务的支持:

      • 如果是插入方法:一定只能在hibernate下才支持(Insert into … select)

       @Transactional // 开启事务!通常会放在业务逻辑层上去声明!
       @Modifying  // 通知springdatajpa 是增删改的操作!

测试代码如下:

  1. 创建repository/CustomerRepository

 package com.yykk.repositories;
 ​
 import com.yykk.pojo.Customer;
 import org.springframework.data.jpa.repository.Modifying;
 import org.springframework.data.jpa.repository.Query;
 import org.springframework.data.repository.PagingAndSortingRepository;
 import org.springframework.data.repository.query.Param;
 import org.springframework.transaction.annotation.Transactional;
 ​
 import java.util.List;
 ​
 public interface CustomerRepositories extends PagingAndSortingRepository<Customer,Long>{
 ​
     // 使用JPQL实现增删改查
 ​
     // 查询
     //@Query("from Customer where custName=?1")
     @Query(value = "from Customer where custName=:custName")
     List<Customer> findCustomerByCustName(@Param("custName") String custName);
 ​
     /**
      *  更新操作!
      *  在这里如果没有事务的支持,那么我们会报错!需要定义在业务逻辑层里面!
      *  @Modifying  // 通知springdatajpa 是增删改的操作!
      */
     @Query("update Customer c set c.custName=:custName where c.id=:id")
     @Transactional // 开启事务!
     @Modifying  // 通知springdatajpa 是增删改的操作!
     int updateCustomerById(@Param("custName") String custName,@Param("id") Long id);
 ​
     // 删除
     @Query("delete from Customer c  where c.id=?1")
     @Transactional // 开启事务!
     @Modifying  // 通知springdatajpa 是增删改的操作!
     String deleteCustomer(Long id);
 ​
     // 新增 JPQL 默认是不支持的,但是这里是使用的伪插入,底层是hibernate!
     // 通知springdatajpa 是增删改的操作!
     //@Transactional // 开启事务!
     //@Modifying
     //@Query(value = "insert into Customer(custName) select cust_name from Customer where id=?1") //这里推荐使用其他方法insert方法不推荐使用!如果要使用可以使用原生的!这里没有values报错!可以尝试一下拼接!
     //int insertCustomerBySelect(Long id);
 ​
     // 原生SQL查询
     // 在这个查询中写成custName之后就报错!
     @Query(value = "select * FROM tb_Customer where cust_name= ? "
             ,nativeQuery = true)
     List<Customer> findCustomerByCustNameBySql(@Param("custName") String custName);
 ​
 }
  1. 测试!

 import com.yykk.config.SpringDataJPAConfig;
 import com.yykk.pojo.Customer;
 import com.yykk.repositories.CustomerRepositories;
 import org.junit.Test;
 import org.junit.runner.RunWith;
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.test.context.ContextConfiguration;
 import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
 ​
 import java.util.List;
 ​
 @ContextConfiguration(classes = SpringDataJPAConfig.class)
 @RunWith(SpringJUnit4ClassRunner.class)
 public class JPQLTest {
 ​
     @Autowired
     CustomerRepositories repositories;
 ​
     /**
      * 查询测试!
      * 因为可能会遇到返回的结果是多个相同的,就使用list接收!
      */
     @Test
     public void testQuery() {
         List<Customer> customer = repositories.findCustomerByCustName("yykk");
         System.out.println(customer);
     }
 ​
     /**
      *  更新操作!
      *  在这里如果没有事务的支持,那么我们会报错!需要定义在业务逻辑层里面!
      */
     @Test
     public void testUpdate() {
         int result = repositories.updateCustomerById("apple", 3L);
         System.out.println(result);
     }
 ​
     @Test
     public void testDelete() {
         String result = repositories.deleteCustomer( 9L);
         System.out.println(result);
     }
 ​
     //@Test
     //public void testInsert() {
     //    int result = repositories.insertCustomerBySelect(1L);
     //    System.out.println(result);
     //}
 ​
     @Test
     public void testQuery_sql() {
 ​
         List<Customer> list = repositories.findCustomerByCustNameBySql("yykk");
         System.out.println(list);
     }
 }

2、规定方法名

  • 只支持查询方法主题关键字(前缀)

    • 只有查询和删除!决定当前方法作用!

  • 查询主题关键字

关键字 描述
find…By`read…Byget…Byquery..Bysearch…Bystream…By 通过查询方法通常返回存储库类型、CollectionStreamable子类型或结果包装器,例如:PageGeoResults或任何其他特定于商店的结果包装器。可用于 findBy…findMyDomainTypeBy…
exists…By 存在投影,通常返回 boolean结果
count…By 计数投影返回数字结果。
delete…By、remove…By 删除查询方法返回无结果(void)或删除计数。
…First<number>……Top<number>… 将查询结果限制为第一个<number>结果。此关键字可以出现在主题的find(和其他关键字)和之间的任何位置by
…Distinct… 使用不同的查询仅返回唯一的结果。查询特定与商店的文档是否支持该功能。此关键字可以出现在主题的 find(和其他关键字)和之间的任意位置 by
  • 支持的查询方法谓词关键字和修饰符

    • 决定查询条件

Keyword Sample JPQL snippet
And findByNameAndPwd where name= ? and pwd =?
Or findByNameOrSex where name= ? or sex=?
Is,Equals findById,findByIdEquals where id= ?
Between findByIdBetween where id between ? and ?
LessThan findByIdLessThan where id < ?
LessThanEquals findByIdLessThanEquals where id <= ?
GreaterThan findByIdGreaterThan where id > ?
GreaterThanEquals findByIdGreaterThanEquals where id > = ?
After findByIdAfter where id > ?
Before findByIdBefore where id < ?
IsNull findByNameIsNull where name is null
isNotNull,NotNull findByNameNotNull where name is not null
Like findByNameLike where name like ?
NotLike findByNameNotLike where name not like ?
StartingWith findByNameStartingWith where name like ‘?%’
EndingWith findByNameEndingWith where name like ‘%?’
Containing findByNameContaining where name like ‘%?%’
OrderBy findByIdOrderByXDesc where id=? order by x desc
Not findByNameNot where name <> ?
In findByIdIn(Collection<?> c) where id in (?)
NotIn findByIdNotIn(Collection<?> c) where id not in (?)
TRUE findByAaaTue where aaa = true
FALSE findByAaaFalse where aaa = false
IgnoreCase findByNameIgnoreCase where UPPER(name)=UPPER(?)
top findTop10 top 10/where ROWNUM <=10
Distinct findDistinctByLastnameAndFirstname select distinct … where x.lastname = ?1 and x.firstname = ?2

1、repository/CustomerMethodNameRepositories

 package com.yykk.repositories;
 ​
 import com.yykk.pojo.Customer;
 import org.springframework.data.jpa.repository.Modifying;
 import org.springframework.data.repository.PagingAndSortingRepository;
 import org.springframework.transaction.annotation.Transactional;
 ​
 import java.util.List;
 ​
 public interface CustomerMethodNameRepositories extends PagingAndSortingRepository<Customer,Long>{
 ​
     List<Customer> findByCustName(String custName);
 ​
     boolean existsByCustName(String custName);
 ​
     @Transactional
     @Modifying
     int deleteByid(Long id);
 ​
     List<Customer> findByCustNameLike(String custName);
 ​
 }

2、测试!

 import com.yykk.config.SpringDataJPAConfig;
 import com.yykk.pojo.Customer;
 import com.yykk.repositories.CustomerMethodNameRepositories;
 import org.junit.Test;
 import org.junit.runner.RunWith;
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.test.context.ContextConfiguration;
 import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
 ​
 import java.util.List;
 ​
 @ContextConfiguration(classes = SpringDataJPAConfig.class)
 @RunWith(SpringJUnit4ClassRunner.class)
 public class MethodName {
 ​
     @Autowired
     CustomerMethodNameRepositories repository;
 ​
     @Test
     public void list(){
         List<Customer> list = repository.findByCustName("yykk");
         System.out.println(list);
     }
 ​
     @Test
     public void exists(){
         boolean exists = repository.existsByCustName("yykk");
         System.out.println(exists);
     }
 ​
     @Test
     public void delete(){
         int del = repository.deleteByid(12L);
         System.out.println(del);
     }
 ​
     @Test
     public void like(){
         List<Customer> list = repository.findByCustNameLike("y%");
         System.out.println(list);
     }
 }

这里的都是静态的固定查询,对于动态的查询要根据以下的这几种方法!

3、通过Query by Example

  • 只支持查询

    • 不支持嵌套或分组的属性约束,如firstname = ?0 or(firstname = ? 1 and lastname = ? 2)

    • 只支持字符串 start/contains/ends/regex 匹配和其他属性类型的精确匹配。

实现:

1、将Repository继承QueryByExampleExecutor

 package com.yykk.repositories;
 ​
 import com.yykk.pojo.Customer;
 import org.springframework.data.repository.PagingAndSortingRepository;
 import org.springframework.data.repository.query.QueryByExampleExecutor;
 ​
 public interface CustomerQBERepositories extends PagingAndSortingRepository<Customer, Long>
         , QueryByExampleExecutor<Customer> {
 ​
 }

2、测试代码!

 import com.yykk.config.SpringDataJPAConfig;
 import com.yykk.pojo.Customer;
 import com.yykk.repositories.CustomerQBERepositories;
 import org.junit.Test;
 import org.junit.runner.RunWith;
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.data.domain.Example;
 import org.springframework.data.domain.ExampleMatcher;
 import org.springframework.test.context.ContextConfiguration;
 import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
 ​
 @ContextConfiguration(classes = SpringDataJPAConfig.class)
 @RunWith(SpringJUnit4ClassRunner.class)
 public class QueryByExampleExecutor {
 ​
     @Autowired
     CustomerQBERepositories repository;
 ​
     /**
      * 简单实例:客户名称,客户地址动态查询!
      */
     @Test
     public void list() {
         Customer customer = new Customer();
         customer.setCustName("yykk");
         customer.setCustAddress("上海");
 ​
         // 通过 Example构造查询条件!
         Example<Customer> example = Example.of(customer);
 ​
         Iterable<Customer> all = repository.findAll(example);
         System.out.println(all);
     }
 ​
     /**
      * 通过匹配器,进行条件的限制!
      * 简单实例:客户名称,客户地址动态查询!
      */
     @Test
     public void test() {
         Customer customer = new Customer();
         customer.setCustName("kk");
         customer.setCustAddress("HAI");
 ​
         // 通过匹配器对条件行为进行设置!
         ExampleMatcher matcher = ExampleMatcher.matching()
                 .withIgnorePaths("custName")  // 设置忽略的属性!
                 //.withIgnoreCase("cust_address")   // 设置忽略大小写,默认不写就是全部属性的设置!
                 //.withStringMatcher(ExampleMatcher.StringMatcher.ENDING); // 对字符串进行结尾匹配
                 .withMatcher("cust_address",m -> m.endsWith().ignoreCase(true));  // 针对单个条件进行设置,会使withIgnoreCase失效!
                 //.withMatcher("custAddress", ExampleMatcher.GenericPropertyMatchers.endsWith());
         // 通过 Example构造查询条件!
         Example<Customer> example = Example.of(customer,matcher);
 ​
         Iterable<Customer> list = repository.findAll(example);
         System.out.println(list);
     }
 ​
 ​
 }

 

4、通过Specifications

之前使用Query by Example只能针对字符串进行条件设置,那如果希望对所有类型支持,可以使用Specifcations

实现

1、继承接口

 package com.yykk.repositories;
 ​
 import com.yykk.pojo.Customer;
 import org.springframework.data.jpa.repository.JpaSpecificationExecutor;
 import org.springframework.data.repository.PagingAndSortingRepository;
 ​
 public interface CustomerSpecificationsRepositories extends PagingAndSortingRepository<Customer,Long>
         ,JpaSpecificationExecutor<Customer> {
     
 }
  • root:查询哪个表(关联查询)= from

  • CriteriaQuery:查询哪些字段,排序是什么 = 组合(order by 、where)

  • CriteriaBuilder:条件之间是什么关系,如何生成一个查询条件,每一个查询条件是什么类型(> between in …)= where

  • Predicate(Expression):每一条查询条件的详细描述

2、测试!

 import com.yykk.config.SpringDataJPAConfig;
 import com.yykk.pojo.Customer;
 import com.yykk.repositories.CustomerSpecificationsRepositories;
 import org.junit.Test;
 import org.junit.runner.RunWith;
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.data.jpa.domain.Specification;
 import org.springframework.test.context.ContextConfiguration;
 import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
 import org.springframework.util.StringUtils;
 ​
 import javax.persistence.EntityManager;
 import javax.persistence.criteria.*;
 import java.util.ArrayList;
 import java.util.List;
 ​
 @ContextConfiguration(classes = SpringDataJPAConfig.class)
 @RunWith(SpringJUnit4ClassRunner.class)
 public class SpecificationsTest {
 ​
     @Autowired
     CustomerSpecificationsRepositories repository;
 ​
     @Autowired
     EntityManager entityManager;
 ​
     @Test
     public void test_All(){
         List<Customer> customer = repository.findAll(new Specification<Customer>() {
             @Override
             public Predicate toPredicate(Root<Customer> root, CriteriaQuery<?> query, CriteriaBuilder criteriaBuilder) {
 ​
                 // root from Customer ,获取列
                 // CriteriaBuilder 设置各种条件 (< > in ...)
                 // query 组合(order by ,where )
 ​
                 return null;
             }
         });
     }
 ​
     /**
      * 查询客户范围(in)
      * id > 大于
      * 地址:精确
      */
     @Test
     public void test_Coll(){
         List<Customer> customer = repository.findAll(new Specification<Customer>() {
             @Override
             public Predicate toPredicate(Root<Customer> root, CriteriaQuery<?> query, CriteriaBuilder criteriaBuilder) {
 ​
                 // root from Customer ,获取列
                 // CriteriaBuilder 设置各种条件 (< > in ...)
                 // query 组合(order by ,where )
 ​
                 Path<Long> id = root.get("id");
                 Path<String> custName = root.get("custName");
                 Path<String> custAddress = root.get("custAddress");
 ​
                 // 参数1:为哪个字段设置条件    参数2:值
                 Predicate address = criteriaBuilder.equal(custAddress, "SHANGHAI");
                 Predicate ids = criteriaBuilder.greaterThan(id, 0L);
                 CriteriaBuilder.In<String> in = criteriaBuilder.in(custName);
                 in.value("yykk").value("张三");
 ​
                 Predicate predicate = criteriaBuilder.and(address, ids,in);
 ​
                 return predicate;
             }
         });
         System.out.println(customer);
     }
 ​
     /**
      * 查询客户范围(in)
      * id > 大于
      * 地址:精确
      * 将这里的值设置为动态的!
      */
     @Test
     public void test_dynamic(){
 ​
         Customer params = new Customer();
         params.setId(0L);
         params.setCustAddress("SHANGHAI");
         params.setCustName("yykk,apple");
 ​
         List<Customer> customer = repository.findAll(new Specification<Customer>() {
             @Override
             public Predicate toPredicate(Root<Customer> root, CriteriaQuery<?> query, CriteriaBuilder criteriaBuilder) {
 ​
                 // root from Customer ,获取列
                 // CriteriaBuilder 设置各种条件 (< > in ...)
                 // query 组合(order by ,where )
 ​
                 Path<Long> id = root.get("id");
                 Path<String> custName = root.get("custName");
                 Path<String> custAddress = root.get("custAddress");
 ​
                 // 参数1:为哪个字段设置条件    参数2:值
                 List<Predicate> list = new ArrayList<>();
                 if (StringUtils.isEmpty(params.getCustAddress())) {
                     list.add(criteriaBuilder.equal(custAddress, "SHANGHAI"));
                 }
                 if (params.getId() > -1) {
                     list.add(criteriaBuilder.greaterThan(id, 0L));
                 }
                 if (StringUtils.isEmpty(params.getCustName())) {
                     CriteriaBuilder.In<String> in = criteriaBuilder.in(custName);
                     in.value("yykk").value("张三");
                     list.add(in);
                 }
 ​
                 Predicate predicate = criteriaBuilder.and(list.toArray(new Predicate[list.size()]));
 ​
                 return predicate;
             }
         });
         System.out.println(customer);
     }
 ​
     /**
      * 查询客户范围(in)
      * id > 大于
      * 地址:精确
      * 将这里的值设置为动态的!
      */
     @Test
     public void test_dynamicx(){
 ​
         CriteriaBuilder builder = entityManager.getCriteriaBuilder();
         CriteriaQuery<Object> query = builder.createQuery();
         Root<Customer> root = query.from(Customer.class);
 ​
         Customer params = new Customer();
         params.setId(0L);
         params.setCustAddress("SHANGHAI");
         params.setCustName("yykk,apple");
 ​
         List<Customer> customer = repository.findAll(new Specification<Customer>() {
             @Override
             public Predicate toPredicate(Root<Customer> root, CriteriaQuery<?> query, CriteriaBuilder criteriaBuilder) {
 ​
                 // root from Customer ,获取列
                 // CriteriaBuilder 设置各种条件 (< > in ...)
                 // query 组合(order by ,where )
 ​
                 // 1、通过root拿到需要设置条件的字段
                 Path<Long> id = root.get("id");
                 Path<String> custName = root.get("custName");
                 Path<String> custAddress = root.get("custAddress");
 ​
                 // 2、通过CriteriaBuilder设置不同的类型条件
                 List<Predicate> list = new ArrayList<>();
                 if (StringUtils.isEmpty(params.getCustAddress())) {
                     // 参数1:为哪个字段设置条件    参数2:值
                     list.add(criteriaBuilder.equal(custAddress, "SHANGHAI"));
                 }
                 if (params.getId() > -1) {
                     list.add(criteriaBuilder.greaterThan(id, 0L));
                 }
                 if (StringUtils.isEmpty(params.getCustName())) {
                     CriteriaBuilder.In<String> in = criteriaBuilder.in(custName);
                     in.value("yykk").value("张三");
                     list.add(in);
                 }
 ​
                 // 3、组合条件!
                 Predicate predicate = criteriaBuilder.and(list.toArray(new Predicate[list.size()]));
 ​
                 Order desc = criteriaBuilder.desc(id);
                 return query.where(predicate).orderBy(desc).getRestriction();
             }
         });
         System.out.println(customer);
     }
 ​
 }

限制:不能分组、聚合函数,需要自己通过entityManager!

 

5、通过Querydsl

官网:http://querydsl.com/

Github:https://github.com/querydsl/querydsl

中文文档:https://github.com/wjw465150/Querydsl-Reference-Guide-Chinese-version

Querydsl 是基于ORM框架或SQL平台上的一个通用查询框架。借助于Querydsl可以在任何支持的ORM框架或者SQL平台上以通用API方式构建查询

JPA是QueryDSL的主要集成技术,是JPQL和Criteria查询的替代方法。目前QueryDSL支持的平台包括JPA、JDO、SQL、MongoDB等等!

QueryDSL扩展让我们以链式方式编写查询方法。该扩展需要一个接口QueryDslPredicateExecutor,它定义了很多查询方法。

1、接口继承了该接口,就可以使用提供的各种方法了!

 package com.yykk.repositories;
 ​
 import com.yykk.pojo.Customer;
 import org.springframework.data.querydsl.QuerydslPredicateExecutor;
 import org.springframework.data.repository.PagingAndSortingRepository;
 ​
 public interface CustomerQueryDSLRepositories extends PagingAndSortingRepository<Customer,Long>
         , QuerydslPredicateExecutor<Customer> {
 ​
 }

2、导入依赖!

 <dependency>
   <groupId>com.querydsl</groupId>
   <artifactId>querydsl-apt</artifactId>
   <version>${querydsl.version}</version>
   <scope>provided</scope>
 </dependency>
 ​
 <dependency>
   <groupId>com.querydsl</groupId>
   <artifactId>querydsl-jpa</artifactId>
   <version>${querydsl.version}</version>
 </dependency>
 ​
 <dependency>
   <groupId>org.slf4j</groupId>
   <artifactId>slf4j-log4j12</artifactId>
   <version>1.6.1</version>
 </dependency>

3、配置Maven APT插件!

 <project>
   <build>
   <plugins>
     <plugin>
       <groupId>com.mysema.maven</groupId>
       <artifactId>apt-maven-plugin</artifactId>
       <version>1.1.3</version>
       <executions>
         <execution>
           <goals>
             <goal>process</goal>
           </goals>
           <configuration>
             <outputDirectory>target/generated-sources/java</outputDirectory>
             <processor>com.querydsl.apt.jpa.JPAAnnotationProcessor</processor>
           </configuration>
         </execution>
       </executions>
     </plugin>
   </plugins>
   </build>
 </project>

JPAAnnotationProcessor 查找使用 javax.persistence.Entity 注解的实体类并为它们生成查询类。

如果您在实体类中使用 Hibernate 注解,则应改用 APT 处理器 com.querydsl.apt.hibernate.HibernateAnnotationProcessor

运行maven clean install,您将得到生成的查询类到 target/generated-sources/java目录下。

如果您使用 Eclipse,请运行 mvn eclipse:eclipse 来更新您的 Eclipse 项目以包含 target/generated-sources/java 作为源文件夹。

简单理解:打开IDEA的项目工程,将我们生成的Q类设置为Sources资源类!

4、测试!

 import com.querydsl.core.Tuple;
 import com.querydsl.core.types.dsl.BooleanExpression;
 import com.querydsl.jpa.impl.JPAQuery;
 import com.querydsl.jpa.impl.JPAQueryFactory;
 import com.yykk.config.SpringDataJPAConfig;
 import com.yykk.pojo.Customer;
 import com.yykk.pojo.QCustomer;
 import com.yykk.repositories.CustomerQueryDSLRepositories;
 import org.junit.Test;
 import org.junit.runner.RunWith;
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.test.context.ContextConfiguration;
 import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
 import org.springframework.util.StringUtils;
 ​
 import javax.persistence.EntityManager;
 import javax.persistence.PersistenceContext;
 import java.util.List;
 ​
 @ContextConfiguration(classes = SpringDataJPAConfig.class)
 @RunWith(SpringJUnit4ClassRunner.class)
 public class QueryDSLTest {
 ​
     @Autowired
     CustomerQueryDSLRepositories repository;
 ​
     // 解决线程安全问题,类似于Autowired!@Bean默认是单例的,JPA就有线程安全问题!
     @PersistenceContext
     EntityManager entityManager;
 ​
 ​
     @Test
     public void test_All(){
         QCustomer customer = QCustomer.customer;
 ​
         // 通过id查找
         BooleanExpression eq = customer.id.eq(1L);
         System.out.println(repository.findOne(eq));
     }
 ​
     /**
      * 查询客户范围(in)
      * id > 大于
      * 地址:精确
      */
     @Test
     public void test_demo(){
         QCustomer customer = QCustomer.customer;
 ​
         // 通过id查找
         BooleanExpression and = customer.custName.in("yykk","apple")
                         .and(customer.id.gt(0L))
                         .and(customer.custAddress.eq("SHANGHAI"));
         System.out.println(repository.findOne(and));
     }
 ​
 ​
     /**
      * 查询客户范围(in)
      * id > 大于
      * 地址:精确
      * 测试动态的!
      */
     @Test
     public void test(){
 ​
         Customer params = new Customer();
         params.setId(0L);
         params.setCustAddress("SHANGHAI");
         params.setCustName("yykk");
 ​
         QCustomer customer = QCustomer.customer;
 ​
         // 初始条件 类似于1=1,永远都成立!
         BooleanExpression expression = customer.isNotNull().or(customer.isNull());
 ​
         expression = params.getId() > -1 ?
                 expression.and(customer.id.gt(params.getId())) : expression;
 ​
         expression = !StringUtils.isEmpty(params.getCustName()) ?
                 expression.and(customer.custName.in(params.getCustName().split(","))) : expression;
 ​
         expression = !StringUtils.isEmpty(params.getCustAddress()) ?
                 expression.and(customer.custAddress.in(params.getCustAddress())) : expression;
 ​
         System.out.println(repository.findAll(expression));
     }
 ​
     /**
      * 自定义列查询、分组
      * 需要使用原生态的方式!(Specification)
      */
     @Test
     public void test_customize(){
         JPAQueryFactory factory = new JPAQueryFactory(entityManager);
 ​
         QCustomer customer = QCustomer.customer;
 ​
         // select id,custName from ...
         // 构建基于QueryDsl的查询
         JPAQuery<Tuple> tupleJPAQuery = factory.select(customer.id, customer.custName)
                 .from(customer)
                 .where(customer.id.eq(1L))
                 .orderBy(customer.id.desc());
 ​
         // 执行查询
         List<Tuple> fetch = tupleJPAQuery.fetch();
 ​
         // 处理返回数据
         for (Tuple tuple : fetch) {
             System.out.println(tuple.get(customer.id));
             System.out.println(tuple.get(customer.custName));
         }
     }
 ​
     /**
      * 自定义列查询、分组
      * 需要使用原生态的方式!(Specification)
      * 通过Repository进行查询,列、表都是固定的!
      */
     @Test
     public void test_customize_list(){
         JPAQueryFactory factory = new JPAQueryFactory(entityManager);
 ​
         QCustomer customer = QCustomer.customer;
 ​
         // select id,custName from ...
         // 构建基于QueryDsl的查询
         JPAQuery<Long> longJPAQuery = factory.select(customer.id.sum())
                 .from(customer)
                 //.where(customer.id.eq(1L))
                 .orderBy(customer.id.desc());
 ​
         // 执行查询
         List<Long> fetch = longJPAQuery.fetch();
 ​
         // 处理返回数据
         for (Long tuple : fetch) {
             System.out.println(tuple);
         }
     }
 }

 

7、多表关联

7.1、多表关联操作

方便我们的开发,导入lombok!

 <dependency>
     <groupId>org.projectlombok</groupId>
     <artifactId>lombok</artifactId>
     <version>1.18.22</version>
 </dependency>

7.2、一对一

客户表——>账户表

 

实现:

1、配置关联关系!

 @OneToOne // 单向关联,一对一!
 @JoinColumn(name = "account_id") // 设置外键的字段名
 private Account account;

pojo/Customer.java

 package com.yykk.pojo;
 ​
 import lombok.AllArgsConstructor;
 import lombok.Data;
 import lombok.NoArgsConstructor;
 ​
 import javax.persistence.*;
 ​
 /**
  * @author yykk
  */
 @Entity // 作为 hibernate实体类
 @Data
 @AllArgsConstructor
 @NoArgsConstructor
 @Table(name = "tb_Customer") // 配置数据库表的名称,实体类中属性和表中字段的映射关系!
 public class Customer {
 ​
     /**
      * @Id:声明主键的配置
      * @GeneratedValue:配置主键的生成策略 strategy
      * GenerationType.IDENTITY :自增,mysql
      * * 底层数据库必须支持自动增长(底层数据库支持的自动增长方式,对id自增)
      * GenerationType.SEQUENCE : 序列,oracle
      * * 底层数据库必须支持序列
      * GenerationType.TABLE : jpa提供的一种机制,通过一张数据库表的形式帮助我们完成主键自增
      * GenerationType.AUTO : 由程序自动的帮助我们选择主键生成策略
      * @Column:配置属性和字段的映射关系 name:数据库表中字段的名称
      */
     @Id
     @GeneratedValue(strategy = GenerationType.IDENTITY)
     private Long id;
 ​
     @Column(name = "cust_name")
     private String custName;//客户名称
 ​
     @Column(name = "cust_source")
     private String custSource;//客户来源
 ​
     @Column(name = "cust_level")
     private String custLevel;//客户级别
 ​
     @Column(name = "cust_industry")
     private String custIndustry;//客户所属行业
 ​
     @Column(name = "cust_phone")
     private String custPhone;//客户的联系方式
 ​
     @Column(name = "cust_address")
     private String custAddress;//客户地址
 ​
     // 单向关联,一对一!
     @OneToOne
     @JoinColumn(name = "account_id") // 设置外键的字段名
     private Account account;
 ​
 }

2、配置关联操作!

pojo/Customer.java

 package com.yykk.pojo;
 ​
 import lombok.AllArgsConstructor;
 import lombok.Data;
 import lombok.NoArgsConstructor;
 ​
 import javax.persistence.*;
 ​
 /**
  * @author yykk
  */
 @Entity // 作为 hibernate实体类
 @Data
 @AllArgsConstructor
 @NoArgsConstructor
 @Table(name = "tb_Customer") // 配置数据库表的名称,实体类中属性和表中字段的映射关系!
 public class Customer {
 ​
     /**
      * @Id:声明主键的配置
      * @GeneratedValue:配置主键的生成策略 strategy
      * GenerationType.IDENTITY :自增,mysql
      * * 底层数据库必须支持自动增长(底层数据库支持的自动增长方式,对id自增)
      * GenerationType.SEQUENCE : 序列,oracle
      * * 底层数据库必须支持序列
      * GenerationType.TABLE : jpa提供的一种机制,通过一张数据库表的形式帮助我们完成主键自增
      * GenerationType.AUTO : 由程序自动的帮助我们选择主键生成策略
      * @Column:配置属性和字段的映射关系 name:数据库表中字段的名称
      */
     @Id
     @GeneratedValue(strategy = GenerationType.IDENTITY)
     private Long id;
 ​
     @Column(name = "cust_name")
     private String custName;//客户名称
 ​
     @Column(name = "cust_source")
     private String custSource;//客户来源
 ​
     @Column(name = "cust_level")
     private String custLevel;//客户级别
 ​
     @Column(name = "cust_industry")
     private String custIndustry;//客户所属行业
 ​
     @Column(name = "cust_phone")
     private String custPhone;//客户的联系方式
 ​
     @Column(name = "cust_address")
     private String custAddress;//客户地址
 ​
     /**
      * 单向关联,一对一!
      * cascade 设置枚举值:关联操作!
      * ALL、PERSIST、MERGE、REMOVE、REFRESH、DETACH
      * fetch:是否设置为懒加载
      *      LAZY 懒加载(知道用到对象才会进行查询!因为不是所有的关联对象,都需要用到)
      *      EAGER 理解加载 (默认)
      *  orphanRemoval:关联移除(通常在修改的时候会用到)
      *      一旦把关联的数据设置为null,或者修改为其他的关联数据,如果想删除数据,就可以设置为true!
      *  optional:设置关联的对象不能为null,默认为true!false不能为null
      *  mappedBy:将外键约束指向另一方维护!(通常在双向关联关系中,会放弃一方的外键约束!)
      *      值 = 另一方的关联属性名!
      */
     @OneToOne(cascade = CascadeType.ALL,fetch = FetchType.LAZY,orphanRemoval = true,optional = false,mappedBy = "customer")
     @JoinColumn(name = "account_id") // 设置外键的字段名
     private Account account;
 ​
 }

pojo/Account.java

 package com.yykk.pojo;
 ​
 import lombok.AllArgsConstructor;
 import lombok.Data;
 import lombok.NoArgsConstructor;
 ​
 import javax.persistence.*;
 ​
 @Entity
 @Table(name = "tb_account")
 @Data
 @AllArgsConstructor
 @NoArgsConstructor
 public class Account {
 ​
     @Id
     @GeneratedValue(strategy = GenerationType.IDENTITY)
     private Long id;
     private String username;
     private String password;
 ​
     @OneToOne
     @JoinColumn(name = "customer_id")
     private Customer customer;
 }

3、测试!

 package com.yykk.test;
 ​
 import com.yykk.config.SpringDataJPAConfig;
 import com.yykk.pojo.Account;
 import com.yykk.pojo.Customer;
 import com.yykk.repositories.CustomerRepository;
 import org.junit.Test;
 import org.junit.runner.RunWith;
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.test.context.ContextConfiguration;
 import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
 import org.springframework.transaction.annotation.Transactional;
 ​
 import java.util.Optional;
 ​
 @ContextConfiguration(classes = SpringDataJPAConfig.class)
 @RunWith(SpringJUnit4ClassRunner.class)
 public class OneToOneTest {
 ​
     @Autowired
     CustomerRepository repository;
 ​
     @Test
     public void test_ins(){
         // 初始化数据
         Account account = new Account();
         account.setUsername("admin");
 ​
         Customer customer = new Customer();
         customer.setCustName("admin");
         customer.setAccount(account);
         account.setCustomer(customer);
 ​
         repository.save(customer);
     }
 ​
     /**
      *  为什么懒加载需要配置事务:
      *  当通过repository调用网查询方法,session就会立即关闭,一旦session关闭就不能查询!
      *  加了事务之后,就能让session直到事务结束完毕后才会关闭!
      */
     @Test
     @Transactional(readOnly = true)
     public void test_sel(){
         Optional<Customer> customer = repository.findById(16L); // 只查询出客户,session就关闭了!
         System.out.println("---------------------");
         System.out.println(customer.get());
     }
 ​
     @Test
     public void test_del(){
         repository.deleteById(1L);
     }
 ​
     @Test
     public void test_upd(){
         Customer customer = new Customer();
         customer.setId(17L);
         customer.setCustName("yykk");
         customer.setAccount(null);
         repository.save(customer);
     }
 }

差异

这两个设置之间的却比在于对 断开关系,例如:当设置地址为null或另一个Address对象。

  • 如果指定了 orphanRemoval = true,则会自动删除断开连接的Address实例,这对于清理很有用没有一个不应该存在的相关对象(例如地址)来自所有者对象(例如员工)的引用。

  • 如果指定 cascade = CascadeType.REMOVE,则不会执行任何自动操作,因为断开关系不是删除操作

 

7.3、一对多

一个客户有多条信息

实现:

1、配置管理关系

@OneToMany

@JoinColumn(name = "customer_id")

 // 一对多
 @OneToMany(cascade = CascadeType.ALL)
 @JoinColumn(name = "customer_id")
 private List<Message> messages;

2、配置关联关系

pojo/Customer.java

 package com.yykk.pojo;
 ​
 import lombok.AllArgsConstructor;
 import lombok.Data;
 import lombok.NoArgsConstructor;
 ​
 import javax.persistence.*;
 import java.util.List;
 ​
 /**
  * @author yykk
  */
 @Entity // 作为 hibernate实体类
 @Data
 @AllArgsConstructor
 @NoArgsConstructor
 @Table(name = "tb_Customer") // 配置数据库表的名称,实体类中属性和表中字段的映射关系!
 public class Customer {
 ​
     /**
      * @Id:声明主键的配置
      * @GeneratedValue:配置主键的生成策略 strategy
      * GenerationType.IDENTITY :自增,mysql
      * * 底层数据库必须支持自动增长(底层数据库支持的自动增长方式,对id自增)
      * GenerationType.SEQUENCE : 序列,oracle
      * * 底层数据库必须支持序列
      * GenerationType.TABLE : jpa提供的一种机制,通过一张数据库表的形式帮助我们完成主键自增
      * GenerationType.AUTO : 由程序自动的帮助我们选择主键生成策略
      * @Column:配置属性和字段的映射关系 name:数据库表中字段的名称
      */
     @Id
     @GeneratedValue(strategy = GenerationType.IDENTITY)
     private Long id;
 ​
     @Column(name = "cust_name")
     private String custName;//客户名称
 ​
     @Column(name = "cust_source")
     private String custSource;//客户来源
 ​
     @Column(name = "cust_level")
     private String custLevel;//客户级别
 ​
     @Column(name = "cust_industry")
     private String custIndustry;//客户所属行业
 ​
     @Column(name = "cust_phone")
     private String custPhone;//客户的联系方式
 ​
     @Column(name = "cust_address")
     private String custAddress;//客户地址
 ​
     /**
      * 单向关联,一对一!
      * cascade 设置枚举值:关联操作!
      * ALL、PERSIST、MERGE、REMOVE、REFRESH、DETACH
      * fetch:是否设置为懒加载
      *      LAZY 懒加载(知道用到对象才会进行查询!因为不是所有的关联对象,都需要用到)
      *      EAGER 理解加载 (默认)
      *  orphanRemoval:关联移除(通常在修改的时候会用到)
      *      一旦把关联的数据设置为null,或者修改为其他的关联数据,如果想删除数据,就可以设置为true!
      *  optional:设置关联的对象不能为null,默认为true!false不能为null
      *  mappedBy:将外键约束指向另一方维护!(通常在双向关联关系中,会放弃一方的外键约束!)
      *      值 = 另一方的关联属性名!
      */
     @OneToOne(cascade = CascadeType.ALL,fetch = FetchType.LAZY,orphanRemoval = true,mappedBy = "customer")
     @JoinColumn(name = "account_id") // 设置外键的字段名
     private Account account;
 ​
     // 一对多
     @OneToMany(cascade = CascadeType.ALL)
     @JoinColumn(name = "customer_id")
     private List<Message> messages;
 ​
 }

pojo/Message.java

 package com.yykk.pojo;
 ​
 import lombok.Data;
 import javax.persistence.*;
 ​
 @Entity
 @Table(name="tb_message")
 @Data
 public class Message {
 ​
     @Id
     @GeneratedValue(strategy = GenerationType.IDENTITY)
     private Long id;
 ​
     // 这里如果手动添加了有参构造,就一定要加无参,否则查询会有问题!
     private String info;
 ​
     public Message() {
     }
 ​
     public Message(String info) {
         this.info = info;
     }
 }

3、代码测试!

 package com.yykk.test;
 ​
 import com.yykk.config.SpringDataJPAConfig;
 import com.yykk.pojo.Customer;
 import com.yykk.pojo.Message;
 import com.yykk.repositories.CustomerRepository;
 import org.junit.Test;
 import org.junit.runner.RunWith;
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.test.context.ContextConfiguration;
 import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
 import org.springframework.transaction.annotation.Transactional;
 ​
 import java.util.ArrayList;
 import java.util.List;
 import java.util.Optional;
 ​
 @ContextConfiguration(classes = SpringDataJPAConfig.class)
 @RunWith(SpringJUnit4ClassRunner.class)
 public class OneToManyTest {
 ​
     @Autowired
     CustomerRepository repository;
 ​
     @Test
     public void test_ins(){
         List<Message> messageList = new ArrayList<>();
         messageList.add(new Message("hello"));
         messageList.add(new Message("word"));
 ​
         Customer customer = new Customer();
         customer.setCustName("yykk");
         customer.setMessages(messageList);
 ​
         repository.save(customer);
     }
 ​
     /**
      * 一对多fetch默认就是LAZY!
      * LAZY过程:
      *  1、findById 指挥查询Customer和其他关联的立即加载!
      *  2、由于输出,会自动调用customer.toString()
      *  LAZY优点:提高查询性能!
      */
     @Test
     @Transactional(readOnly = true)
     public void test_upd(){
         Optional<Customer> id = repository.findById(2L);
         System.out.println("-------------------");
         System.out.println(id);
     }
 ​
     /**
      * 删除操作!设置cascade = CascadeType.ALL | REMOVE
      */
     @Test
     public void test_del(){
         repository.deleteById(4L);
     }
 ​
 }

 

7.4、多对一

实现:

1、配置管理关系

@ManyToOne

@JoinColumn(name = "customer_id")

 // 多对一
 @ManyToOne(cascade = {CascadeType.PERSIST,CascadeType.REMOVE})
 @JoinColumn(name = "customer_id")
 private Customer customer;

2、配置关联关系!

pojo/Customer.java

 package com.yykk.pojo;
 ​
 import lombok.AllArgsConstructor;
 import lombok.Data;
 import lombok.NoArgsConstructor;
 ​
 import javax.persistence.*;
 import java.util.List;
 ​
 /**
  * @author yykk
  */
 @Entity // 作为 hibernate实体类
 @Data
 @AllArgsConstructor
 @NoArgsConstructor
 @Table(name = "tb_Customer") // 配置数据库表的名称,实体类中属性和表中字段的映射关系!
 public class Customer {
 ​
     /**
      * @Id:声明主键的配置
      * @GeneratedValue:配置主键的生成策略 strategy
      * GenerationType.IDENTITY :自增,mysql
      * * 底层数据库必须支持自动增长(底层数据库支持的自动增长方式,对id自增)
      * GenerationType.SEQUENCE : 序列,oracle
      * * 底层数据库必须支持序列
      * GenerationType.TABLE : jpa提供的一种机制,通过一张数据库表的形式帮助我们完成主键自增
      * GenerationType.AUTO : 由程序自动的帮助我们选择主键生成策略
      * @Column:配置属性和字段的映射关系 name:数据库表中字段的名称
      */
     @Id
     @GeneratedValue(strategy = GenerationType.IDENTITY)
     private Long id;
 ​
     @Column(name = "cust_name")
     private String custName;//客户名称
 ​
     @Column(name = "cust_source")
     private String custSource;//客户来源
 ​
     @Column(name = "cust_level")
     private String custLevel;//客户级别
 ​
     @Column(name = "cust_industry")
     private String custIndustry;//客户所属行业
 ​
     @Column(name = "cust_phone")
     private String custPhone;//客户的联系方式
 ​
     @Column(name = "cust_address")
     private String custAddress;//客户地址
 ​
     /**
      * 单向关联,一对一!
      * cascade 设置枚举值:关联操作!
      * ALL、PERSIST、MERGE、REMOVE、REFRESH、DETACH
      * fetch:是否设置为懒加载
      *      LAZY 懒加载(知道用到对象才会进行查询!因为不是所有的关联对象,都需要用到)
      *      EAGER 理解加载 (默认)
      *  orphanRemoval:关联移除(通常在修改的时候会用到)
      *      一旦把关联的数据设置为null,或者修改为其他的关联数据,如果想删除数据,就可以设置为true!
      *  optional:设置关联的对象不能为null,默认为true!false不能为null
      *  mappedBy:将外键约束指向另一方维护!(通常在双向关联关系中,会放弃一方的外键约束!)
      *      值 = 另一方的关联属性名!
      */
     @OneToOne(cascade = CascadeType.ALL,fetch = FetchType.LAZY,orphanRemoval = true,mappedBy = "customer")
     @JoinColumn(name = "account_id") // 设置外键的字段名
     private Account account;
 ​
     // 一对多
     @OneToMany(cascade = CascadeType.ALL)
     @JoinColumn(name = "customer_id")
     private List<Message> messages;
 ​
 }

pojo/Message.java

 package com.yykk.pojo;
 ​
 import lombok.Data;
 import javax.persistence.*;
 ​
 @Entity
 @Table(name="tb_message")
 @Data
 public class Message {
 ​
     @Id
     @GeneratedValue(strategy = GenerationType.IDENTITY)
     private Long id;
 ​
     // 这里如果手动添加了有参构造,就一定要加无参,否则查询会有问题!
     private String info;
 ​
     // 多对一
     @ManyToOne(cascade = {CascadeType.PERSIST,CascadeType.REMOVE})
     @JoinColumn(name = "customer_id")
     private Customer customer;
 ​
     public Message() {
     }
 ​
     public Message(String info, Customer customer) {
         this.info = info;
         this.customer = customer;
     }
 ​
     public Message(String info) {
         this.info = info;
     }
 ​
     @Override
     public String toString() {
         return "Message{" +
                 "id=" + id +
                 ", info='" + info + '\'' +
                 ", customerId=" + customer.getId() +
                 ", customerName=" + customer.getCustName() +
                 '}';
     }
 }

在这里如果你在其中一个里面设置了JoinColumn,那么在另一个中可以不设置,当然设置了也没有问题!

3、设置repository

repositories/MessageRepository

 package com.yykk.repositories;
 ​
 import com.yykk.pojo.Customer;
 import com.yykk.pojo.Message;
 import org.springframework.data.repository.PagingAndSortingRepository;
 ​
 import java.util.List;
 ​
 public interface MessageRepository extends PagingAndSortingRepository<Message,Long> {
 ​
     /**
      * 根据客户id查询所有信息!
      * 通过规定方法名来实现查询,需要通过关联属性来进行匹配!
      * 但是只能通过id来进行匹配
      */
     List<Message> findByCustomer(Customer customer);
 }

4、测试!

 package com.yykk.test;
 ​
 import com.yykk.config.SpringDataJPAConfig;
 import com.yykk.pojo.Customer;
 import com.yykk.pojo.Message;
 import com.yykk.repositories.MessageRepository;
 import org.junit.Test;
 import org.junit.runner.RunWith;
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.test.context.ContextConfiguration;
 import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
 ​
 import java.util.ArrayList;
 import java.util.List;
 ​
 @ContextConfiguration(classes = SpringDataJPAConfig.class)
 @RunWith(SpringJUnit4ClassRunner.class)
 public class ManyToOneTest {
 ​
     @Autowired
     MessageRepository repository;
 ​
     /**
      * 多对一插入:
      *     当插入多的数据的时候,使用多对一的关联关系是更加合适的!
      */
     @Test
     public void test_ins(){
         // 一
         Customer customer = new Customer();
         customer.setCustName("admin");
 ​
         // 多
         List<Message> list = new ArrayList<>();
         list.add(new Message("stay awake",customer));
         list.add(new Message("at all time",customer));
 ​
         repository.saveAll(list);
     }
 ​
     /**
      * 多对一:根据客户id查询对应的所有信息!
      * 这个在一对多实现更简单且合理 ,如果要在这里实现就需要在repository中自定义!
      * 在这里如果出现了栈溢出StackOverFlow,解决方案:
      *    1、在pojo类中对toString方法进行重写!
      */
     @Test
     public void test_query(){
         Customer customer = new Customer();
         customer.setId(1L);
         customer.setCustName("xxx"); // 在这里可以发现,不受其他条件的影响,只会根据id
 ​
         List<Message> messages = repository.findByCustomer(customer);
         // 这里会隐式调用toString()
         System.out.println(messages);
     }
 ​
     @Test
     public void test_delete(){
         Customer customer = new Customer();
         customer.setId(1L);
 ​
         List<Message> messages = repository.findByCustomer(customer);
         // 这里会隐式调用toString()
         repository.deleteAll(messages);
     }
 ​
 }

 

7.5、多对多

1、配置管理关系

@ManyToMany

@JoinColumn(name=“customer_id”)

 /**
  *  单向多对多
  *  中间表需要设置@JoinTable来维护外键(不设置也会自动生成!)
  *  name:指定中间表的表名称
  *  joinColumns:设置本表的外键名称
  *  inverseJoinColumns:设置关联表的外键名称
  * */
 @ManyToMany(cascade = CascadeType.ALL)
 @JoinTable(
         name = "tb_customer_role",
         joinColumns = {@JoinColumn(name = "c_id")},
         inverseJoinColumns = {@JoinColumn(name = "r_id")}
 )
 private Set<Role> roles;

2、配置关联关系!

pojo/Customer.java

 package com.yykk.pojo;
 ​
 import lombok.AllArgsConstructor;
 import lombok.Data;
 import lombok.NoArgsConstructor;
 ​
 import javax.persistence.*;
 import java.util.List;
 import java.util.Set;
 ​
 /**
  * @author yykk
  */
 @Entity // 作为 hibernate实体类
 @Data
 @AllArgsConstructor
 @NoArgsConstructor
 @Table(name = "tb_Customer") // 配置数据库表的名称,实体类中属性和表中字段的映射关系!
 public class Customer {
 ​
     /**
      * @Id:声明主键的配置
      * @GeneratedValue:配置主键的生成策略 strategy
      * GenerationType.IDENTITY :自增,mysql
      * * 底层数据库必须支持自动增长(底层数据库支持的自动增长方式,对id自增)
      * GenerationType.SEQUENCE : 序列,oracle
      * * 底层数据库必须支持序列
      * GenerationType.TABLE : jpa提供的一种机制,通过一张数据库表的形式帮助我们完成主键自增
      * GenerationType.AUTO : 由程序自动的帮助我们选择主键生成策略
      * @Column:配置属性和字段的映射关系 name:数据库表中字段的名称
      */
     @Id
     @GeneratedValue(strategy = GenerationType.IDENTITY)
     private Long id;
 ​
     @Column(name = "cust_name")
     private String custName;//客户名称
 ​
     @Column(name = "cust_source")
     private String custSource;//客户来源
 ​
     @Column(name = "cust_level")
     private String custLevel;//客户级别
 ​
     @Column(name = "cust_industry")
     private String custIndustry;//客户所属行业
 ​
     @Column(name = "cust_phone")
     private String custPhone;//客户的联系方式
 ​
     @Column(name = "cust_address")
     private String custAddress;//客户地址
 ​
     /**
      * 单向关联,一对一!
      * cascade 设置枚举值:关联操作!
      * ALL、PERSIST、MERGE、REMOVE、REFRESH、DETACH
      * fetch:是否设置为懒加载
      *      LAZY 懒加载(知道用到对象才会进行查询!因为不是所有的关联对象,都需要用到)
      *      EAGER 理解加载 (默认)
      *  orphanRemoval:关联移除(通常在修改的时候会用到)
      *      一旦把关联的数据设置为null,或者修改为其他的关联数据,如果想删除数据,就可以设置为true!
      *  optional:设置关联的对象不能为null,默认为true!false不能为null
      *  mappedBy:将外键约束指向另一方维护!(通常在双向关联关系中,会放弃一方的外键约束!)
      *      值 = 另一方的关联属性名!
      */
     @OneToOne(cascade = CascadeType.ALL,fetch = FetchType.LAZY,orphanRemoval = true,mappedBy = "customer")
     @JoinColumn(name = "account_id") // 设置外键的字段名
     private Account account;
 ​
     // 一对多
     @OneToMany(cascade = CascadeType.ALL)
     @JoinColumn(name = "customer_id")
     private List<Message> messages;
 ​
     /**
      *  单向多对多
      *  中间表需要设置@JoinTable来维护外键(不设置也会自动生成!)
      *  name:指定中间表的表名称
      *  joinColumns:设置本表的外键名称
      *  inverseJoinColumns:设置关联表的外键名称
      * */
     @ManyToMany(cascade = CascadeType.ALL)
     @JoinTable(
             name = "tb_customer_role",
             joinColumns = {@JoinColumn(name = "c_id")},
             inverseJoinColumns = {@JoinColumn(name = "r_id")}
     )
     private Set<Role> roles;
 ​
 }

pojo/Role.java

 package com.yykk.pojo;
 ​
 import javax.persistence.*;
 ​
 @Entity
 @Table(name = "tb_role")
 public class Role {
 ​
     @Id
     @GeneratedValue(strategy = GenerationType.IDENTITY)
     private Long id;
 ​
     @Column(name = "role_name")
     private String rName;
 ​
 ​
     public Role(String rName) {
         this.rName = rName;
     }
 ​
     public Role(Long id, String rName) {
         this.id = id;
         this.rName = rName;
     }
 ​
     public Role() {
     }
 }

3、创建repository!

 package com.yykk.repositories;
 ​
 import com.yykk.pojo.Role;
 import org.springframework.data.repository.PagingAndSortingRepository;
 ​
 public interface RoleRepository extends PagingAndSortingRepository<Role,Long> {
 }

4、测试!

 package com.yykk.test;
 ​
 import com.yykk.config.SpringDataJPAConfig;
 import com.yykk.pojo.Customer;
 import com.yykk.pojo.Role;
 import com.yykk.repositories.CustomerRepository;
 import com.yykk.repositories.RoleRepository;
 import org.junit.Test;
 import org.junit.runner.RunWith;
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.test.annotation.Commit;
 import org.springframework.test.context.ContextConfiguration;
 import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
 import org.springframework.transaction.annotation.Transactional;
 ​
 import java.util.HashSet;
 import java.util.Optional;
 import java.util.Set;
 ​
 @ContextConfiguration(classes = SpringDataJPAConfig.class)
 @RunWith(SpringJUnit4ClassRunner.class)
 public class ManyToManyTest {
 ​
     @Autowired
     CustomerRepository repository;
 ​
     @Autowired
     RoleRepository repositories;
 ​
     /**
      * 1、如果保存的数据,希望使用已有的,就需要从数据库中查出来!(持久状态)、否则报错:游离状态不能持久化!
      * 2、如果一个业务方法有多个持久化操作,记得加上@Transactional,否则不能共用一个session
      * 3、在单元测试中,如果用到了@Transactional,如果有增删改操作就需要加 @Commit
      * 4、单元测试会认为你的事务方法@Transactional,只是进行测试,不会提交事务,需要单独加上@Commit
      */
     @Test
     @Transactional
     @Commit
     public void test_INSERT() {
         Set<Role> roles = new HashSet<>();
         roles.add(repositories.findById(2L).get());
         roles.add(repositories.findById(3L).get());
 ​
         Customer customer = new Customer();
         customer.setRoles(roles); // 只要设置了role,就需要设置关联操作!
         customer.setCustName("yykk");
 ​
         repository.save(customer);
     }
 ​
     @Test
     @Transactional(readOnly = true)
     public void test_QUERY() {
         System.out.println(repository.findById(5L));
     }
 ​
     /**
      * 多对多其实并不适合删除,应为经常出现数据可能出现和当前这一端关联还在另一端进行关联!
      * 此时进行删除就会出现:ConstraintViolationException
      * 要进行删除,要保证没有额外的另一端数据关联
      */
     @Test
     @Transactional
     @Commit
     public void test_DELETE() {
         Optional<Customer> customer = repository.findById(5L);
         repository.delete(customer.get());
     }
 }

多对多进行删除可能出现的问题!

可以参考官网文档:https://docs.jboss.org/hibernate/orm/5.6/userguide/html_single/Hibernate_User_Guide.html#associations-many-to-many

 

8、乐观锁

乐观锁是属于hibernate!

防止并发修改!

 private @Version Long version;

 

9、审计

如何使用审计功能

首先申明实体类,需要在类上加上注解@EntityListeners(AuditingEntityListener.class),其次在application启动类上添加注解@EnableJpaAuditing,同时在需要的字段上添加@CreatedDate、@CreatedBy、@LastModifiedDate、@LastModifiedBy等注解。

这个时候,在 Jpa.save()方法被调用的时候,时间字段会自动设置并插入数据库,但是CreatedBy和LastModifiedBy并没有赋值,因为需要实现AuditorAware接口来返回你需要插入的值。

1、编写AuditorAware

 import org.springframework.data.domain.AuditorAware;
 import java.util.Optional;
 ​
 /**
  * 监听
  * @CreatedBy
  * @LastModifiedBy
  * 自动注入用户名
  */
 public class AuditorAwareConfig implements AuditorAware<String> {
     @Override
     public Optional<String> getCurrentAuditor() {
         return Optional.empty();
     }
 }
 ​
 ----------------------------------------------
 ​
 /**
  * 这是JavaConfig方式!
  * AuditorAware 返回当前用户
  * @return
  */
 @Bean
 public AuditorAware<String> auditorAware() {
     return new AuditorAware<String>() {
         @Override
         public Optional getCurrentAuditor() {
             // 当前用户 session  redis  springsecurity
             return Optional.of("yykk");
         }
     };
 }

 

2、在实体类上声明@EntityListeners和响应的注解

考虑到所有实体都要声明,都写在 BaseEntityModel 中

 package com.yykk.config;
 ​
 import lombok.Data;
 import org.springframework.data.annotation.CreatedBy;
 import org.springframework.data.annotation.CreatedDate;
 import org.springframework.data.annotation.LastModifiedBy;
 import org.springframework.data.annotation.LastModifiedDate;
 import org.springframework.data.jpa.domain.support.AuditingEntityListener;
 ​
 import javax.persistence.EntityListeners;
 import javax.persistence.MappedSuperclass;
 import javax.persistence.Temporal;
 import javax.persistence.TemporalType;
 import java.io.Serializable;
 import java.util.Date;
 ​
 @MappedSuperclass
 @EntityListeners(AuditingEntityListener.class)
 @Data
 public class BaseEntityModel implements Serializable {
 ​
     private static final long serialVersionUID  = -6163675075289529459L;
 ​
     @CreatedBy
     String createdBy;
 ​
     @LastModifiedBy
     String modifiedBy;
 ​
     @Temporal(TemporalType.TIMESTAMP)
     @CreatedDate
     protected
     Date dateCreated = new Date();
 ​
     @Temporal(TemporalType.TIMESTAMP)
     @LastModifiedDate
     protected
     Date dateModified = new Date();
 }

3、在Application中启用审计@EnableJpaAuditing

 @EnableJpaAuditing

如果要进行使用的话,需要导入aop的依赖,虽然有aop但是报错没有aspects,添加如下:

 <dependency>
     <groupId>org.springframework</groupId>
     <artifactId>spring-aspects</artifactId>
     <version>5.3.15</version>
 </dependency>

经过测试如果你的实体类上面的多个字段使用了@CreatedBy这样的注解,只会有一个生效,也就是说在一次请求中,只会被调用一次!

 

10、原理

10.1、Repository原理

核心机制:动态代理

  1. JdkDynamicAopProxy—invoke

  • 调用 JPA的Repository统一实现

    • SimpleJpaRepository

    • 就是去通过JPA的API完成持久化操作

 

手写代码进行测试:

源码分析:

 package com.yykk.sourcecode;
 ​
 import com.yykk.config.SpringDataJPAConfig;
 import com.yykk.repositories.CustomerRepository;
 import org.springframework.context.annotation.AnnotationConfigApplicationContext;
 import org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean;
 ​
 import javax.persistence.EntityManager;
 import java.lang.reflect.ParameterizedType;
 import java.lang.reflect.Proxy;
 import java.lang.reflect.Type;
 ​
 public class SourceCode {
     public static void main(String[] args) throws ClassNotFoundException {
         // spring上下文  spring容器 --- ioc加载过程:创建所有的bean,报错repository的Bean
         AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(SpringDataJPAConfig.class);
 ​
         //CustomerRepository repository = context.getBean(CustomerRepository.class);
 ​
         //System.out.println(repository.getClass()); // jdk 动态代理
 ​
         //Optional<Customer> byId = repository.findById(7L);
 ​
         //System.out.println(byId.get());
 ​
         // 获得entityManager
         LocalContainerEntityManagerFactoryBean entityManagerFactoryBean = context.getBean(LocalContainerEntityManagerFactoryBean.class);
         EntityManager nativeEntityManager = entityManagerFactoryBean.createNativeEntityManager(null);
 ​
         // 获得当前接口的pojo类
         // getGenericInterfaces() 拿到当前接口的父接口 = PagingAndSortingRepository
         ParameterizedType parameterizedType = (ParameterizedType)CustomerRepository.class.getGenericInterfaces()[0];
         // 拿到接口的泛型 = <Customer,Long>
         Type type = parameterizedType.getActualTypeArguments()[0];
         Class clazz = Class.forName(type.getTypeName());
 ​
         CustomerRepository repository = (CustomerRepository) Proxy.newProxyInstance(
                 CustomerRepository.class.getClassLoader(),
                 new Class[]{CustomerRepository.class},
                 new MyJpaRepository(nativeEntityManager,clazz)
         );
         repository.findById(7L);
     }
 }
 package com.yykk.sourcecode;
 ​
 import javax.persistence.EntityManager;
 import java.lang.reflect.InvocationHandler;
 import java.lang.reflect.Method;
 ​
 public class MyJpaRepository implements InvocationHandler {
 ​
     EntityManager entityManager;
     Class pojoClass;
 ​
     public MyJpaRepository(EntityManager entityManager, Class pojoClass) {
         this.entityManager = entityManager;
         this.pojoClass = pojoClass;
     }
 ​
     // method 当前调用的方法
     // args 当前调用方法的参数
     @Override
     public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
 ​
         // jpa的统一实现类
         MyJpaProxy myJpaProxy = new MyJpaProxy(entityManager, pojoClass);
         Method targetMethod = myJpaProxy.getClass().getMethod(method.getName(), method.getParameterTypes());
 ​
         return targetMethod.invoke(myJpaProxy,args);
     }
 }
 package com.yykk.sourcecode;
 ​
 import org.springframework.data.domain.Page;
 import org.springframework.data.domain.Pageable;
 import org.springframework.data.domain.Sort;
 import org.springframework.data.repository.PagingAndSortingRepository;
 ​
 import javax.persistence.EntityManager;
 import java.util.Optional;
 ​
 public class MyJpaProxy implements PagingAndSortingRepository {
 ​
     EntityManager entityManager;
     Class pojoClass;
 ​
     public MyJpaProxy(EntityManager entityManager, Class pojoClass) {
         this.entityManager = entityManager;
         this.pojoClass = pojoClass;
     }
 ​
     @Override
     public Iterable findAll(Sort sort) {
         return null;
     }
 ​
     @Override
     public Page findAll(Pageable pageable) {
         return null;
     }
 ​
     @Override
     public Object save(Object entity) {
         return null;
     }
 ​
     @Override
     public Iterable saveAll(Iterable entities) {
         return null;
     }
 ​
     @Override
     public Optional findById(Object id) {
         // JPA
         return Optional.of(entityManager.find(pojoClass,id));
     }
 ​
     @Override
     public boolean existsById(Object o) {
         return false;
     }
 ​
     @Override
     public Iterable findAll() {
         return null;
     }
 ​
     @Override
     public Iterable findAllById(Iterable iterable) {
         return null;
     }
 ​
     @Override
     public long count() {
         return 0;
     }
 ​
     @Override
     public void deleteById(Object o) {
 ​
     }
 ​
     @Override
     public void delete(Object entity) {
 ​
     }
 ​
     @Override
     public void deleteAllById(Iterable iterable) {
 ​
     }
 ​
     @Override
     public void deleteAll(Iterable entities) {
 ​
     }
 ​
     @Override
     public void deleteAll() {
 ​
     }
 }
 package com.yykk.sourcecode;
 ​
 import com.yykk.config.SpringDataJPAConfig;
 import com.yykk.pojo.Customer;
 import com.yykk.repositories.CustomerRepository;
 import org.junit.Test;
 import org.junit.runner.RunWith;
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.data.jpa.repository.config.EnableJpaRepositories;
 import org.springframework.test.context.ContextConfiguration;
 import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
 ​
 import java.util.Optional;
 ​
 @EnableJpaRepositories(basePackages = "com.yykk.repositories")
 @ContextConfiguration(classes = SpringDataJPAConfig.class)
 @RunWith(SpringJUnit4ClassRunner.class)
 public class SpringDataJpaSource {
 ​
     @Autowired
     CustomerRepository repository;
 ​
     @Test
     public void test_source() {
         Optional<Customer> id = repository.findById(7L);
         System.out.println(id.get());
     }
 }

源码学习文章:

http://www.dewafer.com/2016/05/09/reading-src-of-spring-data-jpa/

 

10.2、spring整合JPA原理

1、spring怎么管理Repository(怎么创建的Repository的Bean)

  1. @EnableJpaRepositories(basePackages = "com.yykk.repositories")

  2. spring容器启动的时候ioc容器加载 ,根据 com.yykk.repositories 去创建bean

2、spring怎么将动态代理创建bean

 Exceptioninthread"main"org.springframework.beans.factory.NoSuchBeanDefinitionException:No qualifying bean of type ' com.yykk.repositories.CustomerRepository' available
  1. 没有找到bean,解决思路:

    • 应用层面:是不是配置不正确,配置正确,排除!

    • 底层层面:是不是spring底层扫描-排除。(Bean—>对象)

spring data JPA实现原理:https://zhuanlan.zhihu.com/p/85849213?ivk_sa=1024320u

 

11、springboot整合JPA

1、导入依赖

 <dependencies>
     <dependency>
         <groupId>org.projectlombok</groupId>
         <artifactId>lombok</artifactId>
     </dependency>
     <dependency>
         <groupId>org.springframework.boot</groupId>
         <artifactId>spring-boot-starter-data-jpa</artifactId>
     </dependency>
     <dependency>
         <groupId>org.springframework.boot</groupId>
         <artifactId>spring-boot-starter-web</artifactId>
     </dependency>
 ​
     <dependency>
         <groupId>mysql</groupId>
         <artifactId>mysql-connector-java</artifactId>
         <scope>runtime</scope>
     </dependency>
     <dependency>
         <groupId>org.springframework.boot</groupId>
         <artifactId>spring-boot-starter-test</artifactId>
         <scope>test</scope>
         <exclusions>
             <exclusion>
                 <groupId>org.junit.vintage</groupId>
                 <artifactId>junit-vintage-engine</artifactId>
             </exclusion>
         </exclusions>
     </dependency>
 </dependencies>

2、配置文件

 # 应用名称
 spring.application.name=04-springdata-jpa-springboot
 # 数据库驱动:
 spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver
 # 数据源名称
 spring.datasource.name=defaultDataSource
 # 数据库连接地址
 spring.datasource.url=jdbc:mysql://localhost:3306/spring_data?serverTimezone=UTC
 # 数据库用户名&密码:
 spring.datasource.username=root
 spring.datasource.password=123456
 # 应用服务 WEB 访问端口
 server.port=8080
 ​
 ​
 spring.jpa.show-sql=true
 spring.jpa.database=mysql
 spring.jpa.database-platform=mysql
 spring.jpa.hibernate.ddl-auto=update
 spring.jpa.properties.hibernate.dialect=org.hibernate.dialect.MySQL57Dialect

3、创建实体类

 package com.yykk.pojo;
 ​
 import lombok.AllArgsConstructor;
 import lombok.Data;
 import lombok.NoArgsConstructor;
 import org.springframework.data.jpa.domain.support.AuditingEntityListener;
 ​
 import javax.persistence.*;
 ​
 /**
  * @author yykk
  */
 @Entity // 作为 hibernate实体类
 @Data
 @AllArgsConstructor
 @NoArgsConstructor
 @Table(name = "tb_Customer") // 配置数据库表的名称,实体类中属性和表中字段的映射关系!
 @EntityListeners(AuditingEntityListener.class)
 public class Customer {
 ​
     /**
      * @Id:声明主键的配置
      * @GeneratedValue:配置主键的生成策略 strategy
      * GenerationType.IDENTITY :自增,mysql
      * * 底层数据库必须支持自动增长(底层数据库支持的自动增长方式,对id自增)
      * GenerationType.SEQUENCE : 序列,oracle
      * * 底层数据库必须支持序列
      * GenerationType.TABLE : jpa提供的一种机制,通过一张数据库表的形式帮助我们完成主键自增
      * GenerationType.AUTO : 由程序自动的帮助我们选择主键生成策略
      * @Column:配置属性和字段的映射关系 name:数据库表中字段的名称
      */
     @Id
     @GeneratedValue(strategy = GenerationType.IDENTITY)
     private Long id;
 ​
     @Column(name = "cust_name")
     private String custName;//客户名称
 ​
     @Column(name = "cust_source")
     private String custSource;//客户来源
 ​
     @Column(name = "cust_level")
     private String custLevel;//客户级别
 ​
     @Column(name = "cust_industry")
     private String custIndustry;//客户所属行业
 ​
     @Column(name = "cust_phone")
     private String custPhone;//客户的联系方式
 ​
     @Column(name = "cust_address")
     private String custAddress;//客户地址
 ​
 ​
 }

4、Repositories/CustomerRepository

 package com.yykk.Repositories;
 ​
 import com.yykk.pojo.Customer;
 import org.springframework.data.repository.PagingAndSortingRepository;
 ​
 public interface CustomerRepository extends PagingAndSortingRepository<Customer,Long> {
 }

5、service、serviceImpl

 package com.yykk.service;
 ​
 import com.yykk.pojo.Customer;
 ​
 public interface CustomerService {
 ​
     Iterable<Customer> getAll();
 }
 package com.yykk.service;
 ​
 import com.yykk.Repositories.CustomerRepository;
 import com.yykk.pojo.Customer;
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.stereotype.Service;
 ​
 @Service
 public class CustomerServiceImpl implements CustomerService{
 ​
     @Autowired
     CustomerRepository repository;
 ​
     @Override
     public Iterable<Customer> getAll() {
         return repository.findAll();
     }
 }

6、controller

 package com.yykk.controller;
 ​
 import com.yykk.pojo.Customer;
 import com.yykk.service.CustomerService;
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.web.bind.annotation.RequestMapping;
 import org.springframework.web.bind.annotation.RestController;
 ​
 @RestController
 public class CustomerController {
 ​
     @Autowired
     CustomerService customerService;
 ​
     @RequestMapping("/all")
     public Iterable<Customer> getAll() {
         return customerService.getAll();
     }
 }

springboot自动配置之jpa原理:https://blog.csdn.net/fengyuyeguirenenen/article/details/124114875

可配置项:

 spring.jpa.show-sql=true
 spring.jpa.database=mysql
 spring.jpa.database-platform=mysql
 spring.jpa.hibernate.ddl-auto=update
 spring.jpa.properties.hibernate.dialect=org.hibernate.dialect.MySQL57Dialect

spring.datasource.xxx

  • spring.datasource.schema: 脚本中创建表的语句存放路径,classpath/db表示在工程的resource层级下的db目录中存放。

  • spring.datasource.data:脚本中初始化数据的语句存放路径。

  • spring.datasource.sql-script-encoding:设置脚本的编码,默认常用设置为UTF-8。

  • spring.datasource.driver-class-name:配置driver的类名,默认是从JDBC URL中自动探测。

  • spring.datasource.url:配置数据库JDBC连接串。

  • spring.datasource.username:配置数据库连接用户名。

  • spring.datasource.password:配置数据库连接用户名对应的密码。   使用上述方式建表时,spring.jpa.hibernet.ddl-auto设置成none,否则有啥问题,我也没尝试过。这样配置可以避免两种方式一起使用。

spring.jpa.xxx

  • spring.jpa.hibernet.ddl-auto值说明

  1. create: 服务程序重启后,加载hibernate时都会删除上一次服务生成的表,然后根据服务程序中的model(entity)类再重新生成表,这个值慎用,会导致数据库中原表数据丢失。

  2. create-drop :服务服务程序重启后,加载hibernate时根据model(entity)类生成表,当sessionFactory关闭时,创建的表就自动删除。

  3. update:默认常用属性,第一次加载hibernate时根据model(entity)类会自动建立表结构,后面服务程序重启时,加载hibernate会根据model(entity)类自动更新表结构,如果表结构改变了,但是表行仍然存在,不会删除以前的行(对于表结构行只增不减)。

  4. validate :服务程序重启后,每次加载hibernate时,验证创建数据库表结构,只会和数据库中的表进行比较,如果不同,就会报错。不会创建新表,但是会插入新值。

  5. none : 什么也不做。   我们常用的是update这个属性配置。

  • spring.jpa.database 配置数据库类型,我们常用MYSQL数据库,就配置MYSQL(大小写都可)即可;

  • spring.jpa.properties.hibernate.dialect 使用MYSQL5作为数据库访问方言。

 

配置

 spring.jpa.hibernate.ddl-auto
 create ----每次运行该程序,没有表格会新建表格,表内有数据会清空;
 create-drop ----每次程序结束的时候会清空表
 update ---- 每次运行程序,没有表格会新建表格,表内有数据不会清空,只会更新
 validate ---- 运行程序会校验数据与数据库的字段类型是否相同,不同会报错。
 none —禁用 ddl 处理
 spring.jpa.show-sql=true
 #输出出sql
 ​
 spring.jpa.properties.hibernate.format_sql=true
 #格式化SQL,如果不加,SQL输出不换行,不方便查看
 ​
 spring.jpa.properties.hibernate.max_fetch_depth=1
 #hibernate.max_fetch_depth 属性用于为单向关联(一对一或多对一)的外连接抓取(Outer Join Fetch)树设置最大深度。设置单向关联的外连接抓取树的最大深度为 1,
 ​
 spring.jpa.properties.hibernate.hbm2ddl=update
 自动创建|更新|验证数据库表结构。如果不是此方面的需求建议set value=“none”。
 ​
 create:每次加载hibernate时都会删除上一次的生成的表,然后根据你的model类再重新来生成新表,
 哪怕两次没有任何改变也要这样执行,这就是导致数据库表数据丢失的一个重要原因。
 create-drop :每次加载hibernate时根据model类生成表,但是sessionFactory一关闭,表就自动删除。
 update(***):最常用的属性,第一次加载hibernate时根据model类会自动建立起表的结构(前提是先建立好数据库),
 以后加载hibernate时根据 model类自动更新表结构,即使表结构改变了但表中的行仍然存在不会删除以前的行。
 要注意的是当部署到服务器后,表结构是不会被马上建立起来的,是要等 应用第一次运行起来后才会。
 validate :每次加载hibernate时,验证创建数据库表结构,只会和数据库中的表进行比较,不会创建新表,但是会插入新值。
 spring.jpa.properties.hibernate.enable_lazy_load_no_trans=true
 这个配置的意思就是在没有事务的情况下允许懒加载。
 ​
 spring.jpa.properties.hibernate.temp.use_jdbc_metadata_defaults=false

 

原理图

 

img

posted @ 2022-07-20 15:54  nakano_may  阅读(210)  评论(0编辑  收藏  举报