Spring Data 学习 01 :JDBC 访问 MySQL
Spring Data 学习 02 :Spring JdbcTemplate 访问 MySQL
Spring Data 学习 03 :JPA
Spring Data 学习 04:Spring Data JPA
一、什么是Spring Data JPA:
- 是更大的Spring Data家族的一部分
- 对基于JPA的数据访问层的增强支持
- 更易于构建基于使用Spring数据访问技术栈的应用程序
二、Jpa、Hibernate、Spring Data Jpa三者之间的关系
JPA是ORM规范,它仅仅定义了一些接口。Hibernate、TopLink是实现了JPA规范的ORM框架,具体实现了JPA定义的接口。Spring Data JpaJPA 是对JPA规范的再次封装抽象(Repository层的实现)。Spring Data JPA的核心目标之一是减少代码并简化数据访问层
三、Repository
-
Repository是Spring Data的核心接口,不提供任何方法,包含JpaRepository ,PagingAndSortingRepository等子接口。
定义:
public interface Repository{
}
使用:User是实体类,int代表User的id的类型
添加注解和继承Repository效果一样
// @RepositoryDefinition(domainClass = User.class, idClass = int.class)
public interface UserRepository extends Repository{
public User findByName(String name);
}
Repository同Serializable一样是一个空接口,没有包含方法声明的接口被称之为
标记接口
-
CrudRepository: 继承Repository ,提供基本的增删改查
public interface CrudRepository
extends Repository {
S save(S entity);
T findOne(ID primaryKey);
Iterable findAll();
Long count();
void delete(T entity);
boolean exists(ID primaryKey);
}
-
PagingAndSortingRepository ,继承 CrudRepository ,实现 分页、排序:
public interface PagingAndSortingRepository
extends CrudRepository {
Iterable findAll(Sort sort);
Page findAll(Pageable pageable);
}
-
JpaRepository :继承PagingAndSortingRepository ,具有CrudRepository和PagingAndSortingRepository的所有函数,实现JPA的相关规范,例如刷新持久化上下文和批量删除记录
四、根据方法名创建自定义查询
- 实例
public interface PersonRepository extends JpaRepository{
List findByEmailAddressAndLastname(EmailAddress emailAddress, String lastname);
// 启用 distinct 标志(去重:放在前面后面都可以)
List findDistinctPeopleByLastnameOrFirstname(String lastname, String firstname);
List findPeopleDistinctByLastnameOrFirstname(String lastname, String firstname);
// 给独立的属性启用 ignore case(忽略大小写)
List findByLastnameIgnoreCase(String lastname);
// 给所有合适的属性启用 ignore case
List findByLastnameAndFirstnameAllIgnoreCase(String lastname, String firstname);
// 启用 ORDER BY(排序)
List findByLastnameOrderByFirsnameAsc(String lastname);
List findByLastnameOrderByFirsnameDesc(String lastname);
}
-
Spring Data JPA 自定义查询接口约定命名规则:
五、实体类注解
数据库实体类是一个 POJO Bean 对象。
1、常用注解解释:
* @Entity 用来标示这个类是实体类
* @Table 实体类与数据表的映射,通过name确定在表名(默认类名为表名)比如类Users 和表 users;
* @Id 主键注解,表示该字段为主键
* @GeneratedValue 定义主键规则,默认AUTO
* @Column 类属性与表字段映射注解,其中可以设置在字段名,长度等信息
* @ManyToOne 多对一,可以设置数据加载方式等 默认加载方式是EAGER 就是使用left join
* @OneToMany 一对多 默认加载方式是 LAZY 懒加载
* @JoinColumn 与*对*配合使用,用来设置外键名等信息
* @Basic 实体类中会默认为每一个属性加上这个注解,表示与数据表存在关联,
* 没有使用Column注解的类属性会以属性名作为字段名,驼峰命名需要转为_
* @Temporal 对于Date属性的格式化注解,有 TIME,DATE,TIMESTAMP 几个选择
* @Transient 标注该属性不做与表的映射(原因:可能表中没有该属性对应的字段)有该注解,在执行sql语句时,就不会出现该属性,否则会有,若表中没有该字段则会报错。若存在不想与数据表映射的属性,则需要加上该注解,
2、主键上的注解
// 1、id生成策略自动增长
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private String id;
// 2、配置uuid,本来jpa是不支持uuid的,但借用hibernate的方法
@Id
@GeneratedValue(generator = "uuid") 可以实现。
@GenericGenerator(name = "uuid", strategy = "uuid")
private String id;
- UUID可以自动生成一个36字符组成的字符串,UUID是由10个阿拉伯数字加上26个字母组成,8-4-4-4-12的形式组成,例如:4c47cf4a-a55a-4fce-8cd9-024a790714b010。
- 产生UUID的方法:String uuid=UUID.randomUUID().toString();
六、依赖与配置
- gradle:build.gradle
// 添加 Spring Data JPA依赖
implementation 'org.springframework.boot:spring-boot-starter-data-jpa'
// 添加 MySQL连接驱动依赖
runtimeOnly 'mysql:mysql-connector-java'
- maven:pom.xml
org.springframework.boot
spring-boot-starter-web
mysql
mysql-connector-java
runtime
org.springframework.boot
spring-boot-starter-data-jpa
- 配置:application.properties
#通用数据源配置
spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver
spring.datasource.url=jdbc:mysql://127.0.0.1:3306/student?charset=utf8mb4&useSSL=false
spring.datasource.username=root
spring.datasource.password=root
# Hikari 数据源专用配置
spring.datasource.hikari.maximum-pool-size=20
spring.datasource.hikari.minimum-idle=5
# JPA 相关配置
spring.jpa.database-platform=org.hibernate.dialect.MySQL5InnoDBDialect // 在 SrpingBoot 2.0 版本中,Hibernate 创建数据表的时候,默认的数据库存储引擎选择的是 MyISAM ?这个参数是在建表的时候,将默认的存储引擎切换为 InnoDB.
spring.jpa.show-sql = true // 控制台显示sql
spring.jpa.hibernate.ddl-auto=create // 只能开发中使用:删除数据,重新创建表结构。
七、实战操作
1、建数据库student和表student
CREATE TABLE `student` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`name` varchar(255) DEFAULT NULL,
`age` int(11) DEFAULT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=7 DEFAULT CHARSET=utf8;
2、建立数据库实体类
package com.example.demo.Entity;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
@Entity
public class Student {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
private String name;
private int age;
public Long getId() {
return id;
}
public void setId(Long id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
3、创建数据库持久层
package com.example.demo.Repository;
import com.example.demo.Entity.Student;
import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.data.jpa.repository.Modifying;
import org.springframework.data.jpa.repository.Query;
import org.springframework.data.repository.query.Param;
import java.util.List;
public interface StudentRepository extends JpaRepository {
// 利用注解自定义查询预警
@Query("select o from Student o where id = (select max(id) from Student o)") // Student 是对象
public Student getStudentByMaxId();
// 根据索引参数进行查询
@Query("select o from Student o where o.name=?1 and o.age=?2")
public ListqueryParams(String name, int age);
// 根据别名参数进行查询
@Query("select o from Student o where o.name=:name and o.age=:age")
public ListqueryParams2(@Param("name")String name,@Param("age") int age);
// 原生SQL语句查询
@Query(nativeQuery = true,value = "select count(1) from student") // student是表名
public long getCount();
// @query注解 修改、新增、删除等,要添加@Modifying注解,要在Service里面处理,并增加事务注解 @Transactional
@Modifying
@Query("update Student o set o.age = :age where o.id = :id")
public void update(@Param("id") Long id, @Param("age") int age);
}
4、创建Service
增删改操作需要事务支持,事务要放在Service层,一个service可能要调用多个dao调用,要保证多个调用在一个事务里。
package com.example.demo.Service;
import com.example.demo.Repository.StudentRepository;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import javax.transaction.Transactional;
@Service
public class StudentService {
@Autowired
private StudentRepository studentRepository;
@Transactional
public void update(Long id, int age){
studentRepository.update(id,age);
}
}
5、对持久层 增删改查等操作 进行单元测试
package com.example.demo;
import com.example.demo.Entity.Student;
import com.example.demo.Repository.StudentRepository;
import com.example.demo.Service.StudentService;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.PageRequest;
import org.springframework.data.domain.Sort;
import org.springframework.test.context.junit4.SpringRunner;
import java.util.ArrayList;
import java.util.List;
@RunWith(SpringRunner.class)
@SpringBootTest
public class DemoApplicationTests {
@Autowired
private StudentRepository studentRepository;
@Autowired
private StudentService studentService;
// 自建操作测试
@Test
public void getByMaxId(){
Student student = studentRepository.getStudentByMaxId();
System.out.println("id:"+student.getId()+"name:"+student.getName()+"age:"+student.getAge());
}
@Test
public void queryParams(){
List students = studentRepository.queryParams("护士", 24);
for (Student student:students){
System.out.println("id:"+student.getId()+"name:"+student.getName()+"age:"+student.getAge());
}
}
@Test
public void queryParams2(){
List students = studentRepository.queryParams("护士", 24);
for (Student student:students){
System.out.println("id:"+student.getId()+"name:"+student.getName()+"age:"+student.getAge());
}
}
@Test
public void getCount(){
long count = studentRepository.getCount();
System.out.println(count);
}
@Test
public void update(){
studentService.update(1L,19);
}
// 自带增加、删除、查询、分页、排序 测试
@Test
public void add() {
// 增加一个
Student student = new Student();
student.setName("老师");
student.setAge(23);
studentRepository.save(student);
}
@Test
public void delete() {
// 增加一个
Student student = new Student();
student.setName("老师");
student.setAge(23);
studentRepository.deleteById(1L);
}
@Test
public void findall(){
// 查询所有
ArrayList all = new ArrayList<>();
for (Student student:studentRepository.findAll()){
System.out.println("id:"+student.getId()+"name:"+student.getName()+"age:"+student.getAge());
}
}
// 分页和排序
@Test
public void testPage(){
Sort sort= new Sort(Sort.Direction.DESC, "id");
PageRequest Request = PageRequest.of(0,5, sort);
Page allPage = studentRepository.findAll(Request);
System.out.println("查询总页数"+allPage.getTotalPages());
System.out.println("查询总记录数"+allPage.getTotalElements());
System.out.println("查询当前是第几页"+(allPage.getNumber()+1));
System.out.println("查询当前页面的内容集合"+allPage.getContent());
System.out.println("查询当前页面的记录数"+allPage.getNumberOfElements());
}
}