引入:
Spring Data是SpringSource基金会下的一个用于简化数据库访问,并支持云服务的开源框架。其主要目标是使得数据库的访问变得方便快捷,并支持map-reduce框架和云计算数据服务。对于拥有海量数据的项目,可以用Spring Data来简化项目的开发。
然而针对不同的数据储存访问使用相对的类库来操作访问。Spring Data中已经为我们提供了很多业务中常用的一些接口和实现类来帮我们快速构建项目,比如分页、排序、DAO一些常用的操作。
今天主要是对Spring Data下的JPA模块进行讲解。
为什么说Spring Data能帮助我们快速构建项目呢,因为Spring Data已经在数据库访问层上帮我们实现了公用功能了,而我们只需写一个接口去继承Spring Data提供给我们接口,便可实现对数据库的访问及操作,类似于spring-orm的TemplateDAO。
----------------------------------------------邪恶的分割-------------------------------------------------------
核心接口:
1 public interface Repository
2
3 }
这个接口只是一个空的接口,目的是为了统一所有Repository的类型,其接口类型使用了泛型,泛型参数中T代表实体类型,ID则是实体中id的类型。
再来看一下Repository的直接子接口CrudRepository中的方法:
01 public interface CrudRepository
02
03 S save(S entity);
04
05 Iterable save(Iterable entities);
06
07 T findOne(ID id);
08
09 boolean exists(ID id);
10
11 Iterable
12
13 Iterable
14
15 long count();
16
17 void delete(ID id);
18
19 void delete(T entity);
20
21 void delete(Iterable extends T> entities);
22
23 void deleteAll();
24 }
此接口中的方法大多是我们在访问数据库中常用的一些方法,如果我们要写自己的DAO类的时候,只需定义个接口来集成它便可使用了。
再来看看Spring Data未我们提供分页和排序的Repository的接口PagingAndSortingRepository:
1 public interface PagingAndSortingRepository
2
3 Iterable
4
5 Page
6 }
这些Repository都是spring-data-commons提供给我们的核心接口,spring-data-commons是Spring Data的核心包。这个接口中为我们提供了数据的分页方法,以及排序方法。[b]看吧,spring-data让我们省了很多心了,一切都按照这个规范进行构造,就连业务系统中常用到的一些操作都为我们考虑到了,而我们只需更用心的去关注业务逻辑层。[/b]spring-data将repository的颗粒度划得很细,其实我觉得spring的框架中将每个类的颗粒度都划得很细,这主要也是为了责任分离。
----------------------------------------------邪恶的分割线------------------------------------------------------
JPA实现:
针对spring-data-jpa又提供了一系列repository接口,其中有JpaRepository和JpaSpecificationExecutor,这2个接口又有什么区别呢,我们分别来看看这2个接口的源码。
JpaRepository.class
01 public interface JpaRepository
02
03 List
04
05 List
06
07 List save(Iterable entities);
08 void flush();
09
10 T saveAndFlush(T entity);
11
12 void deleteInBatch(Iterable
13
14 void deleteAllInBatch();
这个类继承自PagingAndSortingRepository,看其中的方法,可以看出里面的方法都是一些简单的操作,并未涉及到复杂的逻辑。当你在处理一些简单的数据逻辑时,便可继承此接口,看一个小例子吧。本文JPA供应者选择的是Hibernate EntityManager,当然读者们也可以选择其他的JPA供应者,比如EclipseLink、OpenJPA,反正JPA是个标准,在无须修改的情况下便可移植。
先定义一用户实体类User.class:
01 @Entity
02 @Table( name = "spring_data_user" )
03 @PrimaryKeyJoinColumn( name = "id" )
04 public class User extends IdGenerator{
05
06 private static final long serialVersionUID = 1L;
07
08 private String name;
09 private String username;
10 private String password;
11 private String sex;
12 private Date birth;
13 private String address;
14 private String zip;
15
16 //省略getter和setter
17 }
Id生成策略是采用的表生成策略,这里就不贴代码了,spring的配置文件我也就不贴出来了,反正就那些东西,网上一查,遍地都是。后续我会在将demo附上来。
实体类是有了,现在得写一个持久层,这样才能操作数据库啊,现在我们来看一下持久层。IUserDao.class:
1 @Repository("userDao")
2 public interface IUserDao extends JpaRepository
再在spring的配置文件中加上以下代码。
1 2 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 3 xmlns:jpa="http://www.springframework.org/schema/data/jpa" 4 xsi:schemaLocation="http://www.springframework.org/schema/beans 5 http://www.springframework.org/schema/data/jpa 6 http://www.springframework.org/schema/data/jpa/spring-jpa.xsd"> 7 8 9
加上这段后Spring就会将指定包中@Repository的类注册为bean,将bean托管给Spring。这样定义完了就OK了!哦,就这样就可以操作数据库了?
是的,前面我就已经说了,Spring data已经帮我们写好一个实现类了,而简单的操作我们只须这样继承JpaRepository就可以做CRUD操作了。再写个业务类来测试一把吧。由于我用的Cglib来动态代理,所以就不定义接口了,直接定义类UserService.class:
01 @Service("userService")
02 public class UserService {
03
04 @Autowired
05 private IUserDao dao;
06
07 public void save(User user) {
08 dao.save(user);
09 }
10
11 public void delete(Long id) {
12 dao.delete(id);
13 }
14
15 public void update(User user) {
16 dao.saveAndFlush(user);
17 }
18
19 public List
20 return dao.findAll();
21 }
22 }
来写一单元测试。
01 public class UserServiceTest {
02
03 private static ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
04
05 private static UserService userService = (UserService) context.getBean("userService");
06
07 public void saveUser() {
08 StopWatch sw = new StopWatch(getClass().getSimpleName());
09 sw.start("Add a user information.");
10 User u = new User();
11 u.setName("John");
12 u.setSex("Man");
13 u.setUsername("JohnZhang");
14 u.setPassword("123456");
15 u.setBirth(new Date());
16 userService.save(u);
17 sw.stop();
18 System.err.println(sw.prettyPrint());
19 }
20
21 public static void main(String[] args) {
22 UserServiceTest test = new UserServiceTest();
23 test.saveUser();
24 }
25 }
绿了,高兴了,测试通过!
额,都没用Junit怎么会绿呢,开个玩笑。
其余继承下来的操作方法,大家都可以自己测试一下,如没意外,应该都会测试通过。
这只是spring data jpa简单的使用,而往往在项目中这一点功能并不能满足我们的需求。这是当然的,在业务中查询是一件非常头疼的事,毕竟不可能只是对一张表的查询是吧? 其实在业务中往往会涉及到多张表的查询,以及查询时需要的各种条件。当然这不用担心,毕竟这是对JPA的支持,而我们在用JPA原生态API的时候往往可能会把一些个方法写得很凌乱,没得一个具体的规范来写自己的方法在后期维护上肯定会很困难。当然你自己也可以封装一些方法来使用,而当我们使用到Spring Data JPA时,它已经帮助我们完成了这个方法的规范了。
来一起看一下复杂查询时它为我们提供的接口。
JpaSpecificationExecutor.class
01 public interface JpaSpecificationExecutor
02
03 T findOne(Specification
04
05 List
06
07 Page
08
09 List
10
11 long count(Specification
12 }
在这个接口里面出现次数最多的类就是Specification.class,而这个类主要也就是围绕Specification来打造的,Specification.class是Spring Data JPA提供的一个查询规范,而你只需围绕这个规范来设置你的查询条件便可,我们来看一下Specification.class这个接口中有些什么东西。
Specification.class
view sourceprint?1 public interface Specification
2
3 Predicate toPredicate(Root
4 }
只有一个方法toPredicate,而其中的参数大家并不陌生,都是JPA规范中的,ROOT查询中的条件表达式、CriteriaQuery条件查询设计器、CriteriaBuilder条件查询构造器,而我们在使用复杂对象查询时,实现该方法用JPA去构造对象查询便可。
下面来看一个小例子:
1 @Repository("userDao")
2 public interface IUserDao extends JpaSpecificationExecutor
3 }
仍然只是一个空接口,这次继承的是JpaSpecificationExecutor了。
再写一测试用例:查询用户表中name包含Sam的记录,并分页按照birth排倒序
01 public class UserDaoTest {
02
03 private static ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
04
05 private static IUserDao userDao = (IUserDao) context.getBean("userDao");
06
07 public void findBySpecAndPaginate() {
08 Page
09 @Override
10 public Predicate toPredicate(Root
11 root = query.from(User.class);
12 Path
13 return cb.like(nameExp, "%Sam%");
14 }
15
16 }, new PageRequest(1, 5, new Sort(Direction.DESC, new String[] { "birth" })));
17
18 StringBuilder stout = new StringBuilder(" 以下是姓名包含Sam人员信息 : ").append("\n");
19 stout.append("| 序号 | username | password | name | sex | birth |").append("\n");
20 int sortIndex = 1;
21 for (User u : page.getContent()) {
22 stout.append(" | ").append(sortIndex);
23 stout.append(" | ").append(u.getUsername());
24 stout.append(" | ").append(u.getPassword());
25 stout.append(" | ").append(u.getName());
26 stout.append(" | ").append(u.getSex());
27 stout.append(" | ").append(u.getBirth());
28 stout.append(" | \n");
29 sortIndex++;
30 }
31 System.err.println(stout);
32 }
33
34 public static void main(String[] args) {
35 UserDaoTest test = new UserDaoTest();
36 test.findBySpecAndPaginate();
37 }
38 }
当然,这只是一个测试,很简单的一个条件查询方法。你也可以设计复杂的查询来得到自己所需的结果,我这只是写一个很简单的方法来带大家入门。