Spring Data
一、 学习安排
Spring-data、spring-boot、spring-cloud
Spring-data是一个spring提供的数据访问层框架。封装若干中数据服务访问能力。如:spring-data-jpa、spring-data-jdbc、spring-data-redis等。
Spring-data-jpa: 是通过JPA标准规范,底层使用Hibernate框架做为实现,开发的关系型数据库访问模块。
Spring-data-jdbc: 就是底层使用Spring-jdbc实现的关系型数据库访问模块。可以使用Mybatis技术作为底层实现的替代产品。
Spring-data-redis:底层使用jedis实现的访问redis数据服务的模块。
上述三个描述的模块是Spring-data中详细讲解的内容。其他的模块非常多,不可能全部讲解。主要作为介绍和配置简单描述。
二、 Spring Data 简介
官网: projects.spring.io/spring-data
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编程模型的,数据访问技术。
Spring Data可以简单快速的访问关系型数据库、非关系型数据库、map-reduce框架和基于云的数据服务。且Spring Data的开发有这些数据服务厂商和开发人员参与。
总而言之,这是一个数据访问层框架,可以高效的访问现今绝大多数数据存储服务,可以统一数据访问层技术,让开发和维护更加方便。
三、 Spring Data JPA
Java persistence api在spring-data工程中的具体应用。
Spring-data-jpa底层是使用Hibernate实现的。
Spring-data-jpa1.8.x+底层依赖的是Hibernate5.x版本的框架。
课程中使用的是spring-data-jpa1.11.x版本。
底层使用Hibernate实现是因为Hibernate本身就是JPA标准规范的实现框架。Hibernate又是经历了长时间商业项目验证的稳定的数据访问层框架。所以spring-data-jpa选择的是Hibernate作为底层实现。
1 Spring-DATA-JPA框架提供的接口的继承树
Spring-Data-JPA提供了若干的Repository接口,提供很多的关系型数据库访问能力。只要对Repository接口有足够的掌握,不需要写SQL就可以使用数据库的访问操作。
1.1 类图
image.png
1.2 Repository
标记接口。可以继承此接口,根据指定的规则,实现数据查询。
1.2.1 方法名称规则查询
这种查询也是有局限性的。只能做单表查询,且查询结果一定是实体类型对象或实体类型对象的集合。毕竟是一个InvocationHandler实现的模板代码,难以完成绝对的定制化开发。只能实现最基础的通用的功能。互联网商业项目中,有半数以上的查询是单表查询。
命名规则:findBy(关键字)+属性名称(属性名称的首字母大写)+查询条件(首字母大写)
关键字 | 方法命名 | sql where字句 |
And | findByNameAndPwd | where name= ? and pwd =? |
Or | findByNameOrSex | where name= ? or sex=? |
Is,Equal | findById,findByIdEquals | where id= ? |
Between | findByIdBetween | where id between ? and ? |
LessThan | findByIdLessThan | where id < ? |
LessThanEqual | findByIdLessThanEquals | where id <= ? |
GreaterThan | findByIdGreaterThan | where id > ? |
GreaterThanEqual | 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(?) |
1.2.2 @Query注解查询
使用@Query注解描述方法,可以绕过方法命名规则,通过编写JPQL或SQL来实现数据访问。
JPQL:java persistence 通过Hibernate的HQL演变过来的。他和HQL语法及其相似。JPQL语法查询,可以实现投影查询,可以实现表连接查询,毕竟是通过一种查询语法规则实现的查询逻辑。From Users、 select username from Users、from Users u,Roles r where u.xxx = r.xxx。JPQL语法查询的时候,在JPQL(1.11.14)中,传递参数的时候,使用变量命名的方式,通过’:变量名’来定义语法中的变量,方法的参数表中,通过@Param注解,描述方法参数表中的参数和JPQL中变量的对应关系,注解的属性value代表JPQL中变量的名称。
nativeQuery: 通过SQL语法实现数据查询。效率更高的执行逻辑。因为spring-data-jpa不需要实现JPQL->SQL的语法转换。少了JPQL字符串的解析,SQL字符串的拼接等过程,效率相对更高。
语法如下:JPQL或SQL中的变量占位符顺序对应方法参数表中的参数顺序。
public interface IA{ @Query(value="JPQL") List xxx(); @Query(value="SQL", nativeQuery=true) List yyy(); }
public interface IA{ @Query(value="JPQL") List xxx(); @Query(value="SQL", nativeQuery=true) List yyy(); }
1.2.3 @Query注解实现数据写操作
@Query注解是用于描述数据访问语法的,所以可以执行CRUD任意操作。但是JPQL只能执行查询和更新操作,SQL可以执行CRUD任意操作。语法如下:JPQL或SQL中的变量占位符顺序对应方法参数表中的参数顺序。
public interface IA{ @Query(value="JPQL") @Modifying List xxx(); @Query(value="SQL", nativeQuery=true) @Modifying List yyy(); }
public interface IA{ @Query(value="JPQL") @Modifying List xxx(); @Query(value="SQL", nativeQuery=true) @Modifying List yyy(); }
1.3 CrudRepository
定义了CRUD操作的接口,是Repository的子接口。接口中没有定义独立的更新方法,调用新增数据方法(save)的时候,底层会根据方法参数决定是新增数据还是更新数据。因为底层是Hibernate,Hibernate中有方法saveOrUpdate,而spring-data-jpa中提供的save方法底层使用的就是Hibernate中的saveOrUpdate方法。判断依据,就是数据是否存在,就是数据对象的id是否存在。
1.4 PagingAndSortingRepository
定义了分页和排序查询的接口,是CrudRepository的子接口。
1.5 JpaRepository
PagingAndSortingRepository的子接口,提供了对缓存的处理方法。
1.6 JpaSpecificationExecutor
提供了条件查询和分页处理的接口,是一个相对独立的接口。这个接口的使用必须配合Repository接口树中的任意接口。
2 自定义Repository
Spring-data-jpa做开发的时候,一般不会去定义DAO接口的实现类。因为spring-data-jpa会提供一个动态代理对象,动态代理对象的类型是SimpleJpaRepository。
2.1 自定义Repository接口
定义一个Repository接口,不需要继承任何接口,定义需要的自定义方法。
2.2 定义Dao接口
定义一个Dao接口,这个接口是服务代码调用的。Dao接口继承spring-data-jpa提供的Repository接口和自定义的Repository接口。
2.3 定义Dao接口实现
提供一个接口实现类,这个实现类只需要实现自定义Repository接口,但是其类名必须和Dao接口相关,命名规则为"Dao接口名Impl",如:Dao接口为UserDao,实现类命名为UserDaoImpl,实现类实现接口Repository。
符合上述要求的代码spring-data-jpa会自动为Dao创建一个动态代理对象,这个动态代理对象会提供spring-data-jpa定义的Repository相关代码实现,并使用自定义的接口实现类提供自定义Repository接口的方法实现。
类图如下:
image.png
3 Spring-data-jpa正向工程建表
在默认情况下(spring-data-jpa 1.11.x),spring-data-jpa使用正向工程创建的数据库表格,是有默认配置的,其建表语句如下:
create table tb_users (
userid integer not null auto_increment,
userage integer,
username varchar(255),
primary key (userid)
) engine=MyISAM
Spring-data-jpa默认情况下,创建的MySQL数据库表格是MyISAM引擎的表格。这种表格,适合查询,不支持事务,不支持外键约束。
但是,因为其不支持事务,所有有脏数据出现的可能。
可以在配置LocalContainerEntityManagerFactoryBean的时候,增加配置信息,约束建表的引擎。
<!-- Spring整合JPA 配置EntityManagerFactory-->
<bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean"> <property name="dataSource" ref="dataSource"/> <property name="jpaVendorAdapter"> <bean class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter"> <!-- hibernate相关的属性的注入 --> <!-- 配置数据库类型 --> <property name="database" value="MYSQL"/> <!-- 正向工程 自动创建表 --> <property name="generateDdl" value="true"/> <!-- 显示执行的SQL --> <property name="showSql" value="true"/> <!-- 提供数据库特性配置。和数据库的独特功能紧密相关。这个配置是使用Hibernate提供的Dialect来配置的。 --> <property name="databasePlatform" value="org.hibernate.dialect.MySQL5InnoDBDialect" /> </bean> </property> <!-- 扫描实体的包 --> <property name="packagesToScan"> <list> <value>com.bjsxt.pojo</value> </list> </property> </bean>
修改配置后,建表语句如下:
create table tb_users (
userid integer not null auto_increment,
userage integer,
username varchar(255),
primary key (userid)
) engine=InnoDB
是否需要配置数据库特性,由具体业务决定。一般来说,商业项目中,不可能使用正向工程建表。也就是说,配置应该是<property name="generateDdl" value="false"/>
4 关联操作
关联操作在spring-data-jpa中主要通过JPA注解来实现。如:@OneToOne、@OneToMany、@ManyToOne、@ManyToMany、@JoinColumn、@JoinTable等。
详见代码。
四、 Spring Data Redis
spring-data框架中的每个子模块其版本未必一致,毕竟对应不同数据服务的访问层框架,更新时间和周期是不同的。在本案例中,使用的spring-data-redis版本为1.8.14。
spring-data-redis框架的执行需要jackson组件的辅助,建议导入jackson版本为2.7+(对应当前环境中的spring-data-redis版本)。