`

spring data jpa Repository

 
阅读更多
AppleFramework在数据访问控制层采用了Spring Data作为这一层的解决方案,

 

下面就对Spring Data相关知识作一个较为详细的描述。

 

1.Spring Data所解决的问题

 

Spring Data :提供了一整套数据访问层(DAO)的解决方案,

 

致力于减少数据访问层(DAO)的开发量。

 

它使用一个叫作Repository的接口类为基础,

 

它被定义为访问底层数据模型的超级接口。

 

而对于某种具体的数据访问操作,则在其子接口中定义。

 

public interface Repository<T, ID extends Serializable> {

 

}

 

 

 

所有继承这个接口的interface都被spring所管理,此接口作为标识接口,功能就是用来控制domain模型的。

 

Spring Data可以让我们只定义接口,只要遵循spring data的规范,就无需写实现类。

 

 

 

2.什么是Repository?

 

2.1 Repository(资源库):通过用来访问领域对象的一个类似集合的接口,

 

在领域与数据映射层之间进行协调。这个叫法就类似于我们通常所说的DAO,

 

在这里,我们就按照这一习惯把数据访问层叫Repository

 

Spring Data给我们提供几个Repository,基础的Repository提供了最基本的数据访问功能,

 

其几个子接口则扩展了一些功能。它们的继承关系如下:

 

Repository: 仅仅是一个标识,表明任何继承它的均为仓库接口类,方便Spring自动扫描识别

 

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

 

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

 

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

 

JpaSpecificationExecutor: 比较特殊,不属于Repository体系,实现一组JPA Criteria查询相关的方法

 

我们自己定义的XxxxRepository需要继承JpaRepository,

 

这样我们的XxxxRepository接口就具备了通用的数据访问控制层的能力。

 

2.2 JpaRepository 所提供的基本功能

 

2.2.1 CrudRepository<T, ID extends Serializable>:

 

这个接口提供了最基本的对实体类的添删改查操作

 

T save(T entity);//保存单个实体

 

        Iterable<T> save(Iterable<? extends T> entities);//保存集合

 

        T findOne(ID id);//根据id查找实体

 

        boolean exists(ID id);//根据id判断实体是否存在

 

        Iterable<T> findAll();//查询所有实体,不用或慎用!

 

        long count();//查询实体数量

 

        void delete(ID id);//根据Id删除实体

 

        void delete(T entity);//删除一个实体

 

        void delete(Iterable<? extends T> entities);//删除一个实体的集合

 

        void deleteAll();//删除所有实体,不用或慎用!

 

2.2.2 PagingAndSortingRepository<T, ID extends Serializable>

 

这个接口提供了分页与排序功能

 

Iterable<T> findAll(Sort sort);//排序

 

        Page<T> findAll(Pageable pageable);//分页查询(含排序功能)

 

2.2.3 JpaRepository<T, ID extends Serializable>

 

这个接口提供了JPA的相关功能

 

List<T> findAll();//查找所有实体

 

        List<T> findAll(Sort sort);//排序 查找所有实体

 

        List<T> save(Iterable<? extends T> entities);//保存集合

 

        void flush();//执行缓存与数据库同步

 

        T saveAndFlush(T entity);//强制执行持久化

 

void deleteInBatch(Iterable<T> entities);//删除一个实体集合

 

3.Spring data 查询

 

3.1 简单条件查询:查询某一个实体类或者集合

 

按照Spring data 定义的规则,查询方法以find|read|get开头

 

涉及条件查询时,条件的属性用条件关键字连接,要注意的是:条件属性以首字母大写其余字母小写为规定。

 

例如:定义一个Entity实体类

 

class User{

 

private String firstname;

 

private String lastname;

 

 

使用And条件连接时,应这样写:

 

findByLastnameAndFirstname(String lastname,String firstname);

 

条件的属性名称与个数要与参数的位置与个数一一对应

 

其他条件关键字如下表的定义:

 

Table 3.1. Supported keywords inside method names
 

 

 

 

 

 

3.2 使用JPA NamedQueries (标准规范实现)

 

这种查询是标准的JPA规范所定义的,直接声明在Entity实体类上,

 

调用时采用在接口中定义与命名查询对应的method,由Spring Data根据方法名自动完成命名查询的寻找。

 

(1)在Entity实体类上使用@NamedQuery注解直接声明命名查询。

 

@Entity

 

@NamedQuery(name = "User.findByEmailAddress", 

 

  query = "select u from User u where u.emailAddress = ?1")

 

public class User {

 

 

 

}

 

注:定义多个时使用下面的注解

 

@NamedQueries(value = { 

 

                @NamedQuery(name = User.QUERY_FIND_BY_LOGIN,

 

                                        query = "select u from User u where u." + User.PROP_LOGIN

 

                                                + " = :username"),

 

        @NamedQuery(name = "getUsernamePasswordToken",

 

                        query = "select new com.aceona.weibo.vo.TokenBO(u.username,u.password) from User u where u." + User.PROP_LOGIN

 

                            + " = :username")})

 

(2)在interface中定义与(1)对应的方法

 

public interface UserRepository extends JpaRepository<User, Long> {

 

 

 

  List<User> findByLastname(String lastname);

 

 

 

  User findByEmailAddress(String emailAddress);

 

}

 

3.3 使用@Query自定义查询(Spring Data提供的)

 

这种查询可以声明在Repository方法中,摆脱像命名查询那样的约束,

 

将查询直接在相应的接口方法中声明,结构更为清晰,这是Spring data的特有实现。

 

例如:

 

public interface UserRepository extends JpaRepository<User, Long> {

 

 

 

  @Query("select u from User u where u.emailAddress = ?1")

 

  User findByEmailAddress(String emailAddress);

 

}

 

3.4 @Query与 @Modifying 执行更新操作

 

这两个annotation一起声明,可定义个性化更新操作,例如只涉及某些字段更新时最为常用,示例如下:

 

@Modifying

 

@Query("update User u set u.firstname = ?1 where u.lastname = ?2")

 

int setFixedFirstnameFor(String firstname, String lastname);

 

 

 

3.5 索引参数与命名参数

 

(1)索引参数如下所示,索引值从1开始,查询中 ”?X” 个数需要与方法定义的参数个数相一致,并且顺序也要一致

 

@Modifying

 

@Query("update User u set u.firstname = ?1 where u.lastname = ?2")

 

int setFixedFirstnameFor(String firstname, String lastname);

 

 

 

(2)命名参数(推荐使用这种方式)

 

可以定义好参数名,赋值时采用@Param("参数名"),而不用管顺序。如下所示:

 

public interface UserRepository extends JpaRepository<User, Long> {

 

 

 

  @Query("select u from User u where u.firstname = :firstname or u.lastname = :lastname")

 

  User findByLastnameOrFirstname(@Param("lastname") String lastname,

 

                                 @Param("firstname") String firstname);

 

}

 

 

 

4. Transactionality(事务)

 

4.1 操作单个对象的事务

 

Spring Data提供了默认的事务处理方式,即所有的查询均声明为只读事务,对于持久化,更新与删除对象声明为有事务。

 

参见org.springframework.data.jpa.repository.support.SimpleJpaRepository<T, ID>

 

@org.springframework.stereotype.Repository

 

@Transactional(readOnly = true)

 

public class SimpleJpaRepository<T, ID extends Serializable> implements JpaRepository<T, ID>,

 

                JpaSpecificationExecutor<T> {

 

……

 

@Transactional

 

        public void delete(ID id) {

 

 

 

                delete(findOne(id));

 

        }

 

……

 

}

 

对于自定义的方法,如需改变spring data提供的事务默认方式,可以在方法上注解@Transactional声明

 

 

 

4.2 涉及多个Repository的事务处理

 

进行多个Repository操作时,也应该使它们在同一个事务中处理,按照分层架构的思想,这部分属于业务逻辑层,因此,需要在Service层实现对多个Repository的调用,并在相应的方法上声明事务。

 

例如:

 

@Service(“userManagement”)

 

class UserManagementImpl implements UserManagement {

 

 

 

  private final UserRepository userRepository;

 

  private final RoleRepository roleRepository;

 

 

 

  @Autowired

 

  public UserManagementImpl(UserRepository userRepository, 

 

    RoleRepository roleRepository) {

 

    this.userRepository = userRepository;

 

    this.roleRepository = roleRepository;

 

  }

 

 

 

  @Transactional

 

  public void addRoleToAllUsers(String roleName) {

 

 

 

    Role role = roleRepository.findByName(roleName);

 

 

 

    for (User user : userRepository.readAll()) {

 

      user.addRole(role);

 

      userRepository.save(user);

 

    }

 

}

 

 

 

5.关于DAO层的规范

 

5.1对于不需要写实现类的情况:

 

定义XxxxRepository 接口并继承JpaRepository接口,

 

如果Spring data所提供的默认接口方法不够用,可以使用@Query在其中定义个性化的接口方法。

 

5.2对于需要写实现类的情况:

 

定义XxxxDao 接口并继承com.aceona.appleframework.persistent.data.GenericDao

 

书写XxxxDaoImpl实现类并继承com.aceona.appleframework.persistent.data.GenericJpaDao,

 

同时实现XxxxDao接口中的方法

 

 

 

在Service层调用XxxxRepository接口与XxxxDao接口完成相应的业务逻辑

 

分享到:
评论

相关推荐

    Spring Data JPA中文文档[1.4.3]_springdatajpa_erlang_waitxpf_

    开发者可以通过 `EntityManager` 和 `EntityManagerFactory` 进行这些操作,但 Spring Data JPA 提供的 Repository 接口让这些操作更加简洁。 3. **Query Methods**:Spring Data JPA 支持通过方法名自动转换为 JPA...

    Spring Data JPA 笔记

    通过继承`JpaRepository`或自定义Repository接口,可以轻松实现数据访问。例如,`findById`、`save`等方法都由Spring Data JPA自动提供。 3. **Querydsl**:Spring Data JPA可以与Querydsl结合使用,提供更强大的...

    Spring Boot JpaRepository知识学习(Spring Data JPA)

    Spring Boot JpaRepository知识学习(Spring Data JPA) Spring Boot与JpaRepository是Java开发中常见的组合,它们简化了数据库操作,使得开发者可以快速构建基于Spring的应用。本文将深入探讨Spring Data JPA,它...

    spring注解+spring data jpa文档+JPA文档.rar

    Spring Data JPA提供了强大的Repository抽象,允许开发者以声明式的方式定义数据操作,如查询方法。只需要在接口上定义方法名,Spring Data JPA就能自动生成对应的SQL语句。例如,`findAll()`会执行SELECT ALL查询,...

    Spring Data JPA从入门到精通

    'SpringDataJPA从入门到精通'分为12章 内容包括整体认识JPA、JPA基础查询方法、定义查询方法、注解式查询方法、@Entity实例里面常用注解详解、JpaRepository扩展详解、JPA的MVC扩展REST支持、DataSource的配置、乐观...

    Spring Data JPA的优点和难点.pdf

    - Spring Data JPA通过提供自动化的 Repository 实现,减少了大量手动编写SQL和DAO层代码的工作。开发者只需定义Repository接口,通过简单的查询方法名即可实现复杂的数据库查询。例如,`...

    spring data jpa 教程

    JpaRepository 是 Spring Data JPA 中提供的一个基础接口,它扩展自 PagingAndSortingRepository,并且增加了一些额外的、常用的操作方法。通过实现 JpaRepository 接口,开发者可以得到一系列的预定义方法,例如...

    手动创建 SpringMvc +SpringDataJpa+Hibernate+ freemarker mavenProject+ 环境切换 webDemo

    5. JpaRepository接口:定义数据库操作,Spring Data JPA会自动生成实现。 6. Service层接口及实现:业务逻辑处理,调用JpaRepository接口进行数据操作。 7. FreeMarker模板文件(`.ftl`):用于生成视图。 8. 控制...

    SpringDataJpa开发--继承JpaRepository实现简单条件查询

    `JpaRepository`是Spring Data JPA提供的一种高级抽象,它允许我们以声明式的方式处理CRUD操作,以及执行基本的查询。 首先,我们需要理解`JpaRepository`接口的作用。它是由Spring Data JPA提供的一个基础接口,...

    Spring Data JPA.zip

    定义接口继承 `JpaRepository` 或自定义接口,并注入到 Spring 容器中。通过接口的方法即可完成数据的增删改查。 **6. 查询方法命名规则** Spring Data JPA 支持根据方法名自动转换为 SQL 查询,例如 `findAll()`、...

    Spring Data JPA Demo

    Spring Data JPA 基于Repository 模式,我们定义一个接口,继承`JpaRepository` 或`PagingAndSortingRepository`,然后可以不用编写实现类,Spring Data JPA 将自动为这些接口生成实现。 5. **自定义查询** 如果...

    Spring Data JPA中文文档[1.4.3].zip

    6. **分页和排序**:Spring Data JPA支持在查询结果中进行分页和排序,可以方便地通过Repository接口实现。 7. **存储过程**:Spring Data JPA也支持调用数据库存储过程,并将其结果转换为Java对象。 8. **JPA实体...

    spring data jpa简单案例

    import org.springframework.data.jpa.repository.JpaRepository; public interface UserRepository extends JpaRepository, Long&gt; { } ``` 无需编写任何实现,Spring Data JPA 会自动处理这些操作。 ### 5. 使用...

    springBoot整合springData JPA

    **SpringBoot整合SpringData JPA** 是一个现代Java开发中的常见技术栈,它结合了Spring Boot的便捷性和Spring Data JPA的数据访问效率。Spring Boot简化了应用的初始搭建以及配置,而Spring Data JPA则是Spring ...

    Spring Boot整合SpringDataJPA

    我们可以通过继承`JpaRepository`或`CrudRepository`来创建自定义的Repository,无需编写任何DAO层的代码。例如,假设我们有一个`User`实体,我们可以定义一个`UserRepository`: ```java public interface ...

    其实spring data jpa比mybatis更好用.zip_JPA mybatis

    4. **维护**:Spring Data JPA的Repository模式降低了代码的耦合度,而MyBatis的DAO层需要手动维护。 综上所述,Spring Data JPA在许多方面提供了便利,尤其是对于快速开发和维护,但在特定场景下,如需要高度定制...

    spring学习:spring data jpa

    5. 创建Repository接口:定义Repository接口并继承自Spring Data JPA提供的接口,如`JpaRepository`或`PagingAndSortingRepository`。 6. 使用Repository:在服务层,通过@Autowired注解注入Repository接口,然后...

    springdatajpa.pdf

    SpringDataJPA是Spring框架中用于简化数据持久层操作的一个模块,它基于Java持久层API(Java Persistence API,JPA)标准。在实际的项目开发中,SpringDataJPA能够极大程度上减少我们对于数据访问层代码的编写工作。...

Global site tag (gtag.js) - Google Analytics