2019独角兽企业重金招聘Python工程师标准>>>
Cliff 发布于 10 个月前 0评论 3426浏览 springjavaspringbootjpaspring-data
Spring Data JPA中的Repository是接口,是JPA根据方法名帮我们自动生成的。但很多时候,我们需要为Repository提供一些自定义的实现。今天我们看看如何为Repository添加自定义的方法。
自定义Repository接口
首先我们来添加一个自定义的接口:
- 添加
BaseRepository
接口 BaseRepository
继承了PagingAndSortingRepository
,这样可以保证所有Repository都有基本的增删改查以及分页等方法。- 在
BaseRepository
上添加@NoRepositoryBean
标注,这样Spring Data Jpa在启动时就不会去实例化BaseRepository
这个接口 - 添加
support(String modelType)
方法,表示该Repository的领域对象是否为modelType
类型
@NoRepositoryBean
public interface BaseRepository
extends PagingAndSortingRepository {
boolean support(String modelType);
}
然后,使所有Repository接口都继承BaseRepository
实现BaseRepository接口
定义好自定义的方法后,我们现在通过一个基本的Repository类来实现该方法:
首先添加BaseRepositoryImpl类,继承SimpleJpaRepository类,使其拥有Jpa Repository的基本方法。
我们发现Repository有两个构造函数:
- SimpleJpaRepository(JpaEntityInformation entityInformation, EntityManager entityManager)
- SimpleJpaRepository(Class domainClass, EntityManager em)
这里我们实现第二个构造函数,拿到domainClass
和EntityManager
两个对象。因为我们要实现的是知道某个Repository是否支持某个领域对象的类型,因此在实现构造函数时我们将domainClass
的信息保留下来。
最后实现support
方法,其参数是领域对象的类型,将其和domainClass
对比,如果相等,则该Repository支持该类型的领域对象:
public class BaseRepositoryImpl
extends SimpleJpaRepository
implements BaseRepository {
private final Class domainClass;
public BaseRepositoryImpl(Class domainClass, EntityManager entityManager) {
super(domainClass, entityManager);
this.domainClass = domainClass;
}
@Override
public boolean support(String modelType) {
return domainClass.getName().equals(modelType);
}
}
创建自定义RepositoryFactoryBean
接下来我们来创建一个自定义的RepositoryFactoryBean
来代替默认的RepositoryFactoryBean
。RepositoryFactoryBean
负责返回一个RepositoryFactory
,Spring Data Jpa 将使用RepositoryFactory
来创建Repository具体实现,这里我们用BaseRepositoryImpl
代替SimpleJpaRepository
作为Repository接口的实现。这样我们就能够达到为所有Repository添加自定义方法的目的。
public class BaseRepositoryFactoryBean, T, I extends Serializable>
extends JpaRepositoryFactoryBean {
@Override
protected RepositoryFactorySupport createRepositoryFactory(EntityManager em) {
return new MyRepositoryFactory(em);
}
private static class MyRepositoryFactory extends JpaRepositoryFactory {
private final EntityManager em;
public MyRepositoryFactory(EntityManager em) {
super(em);
this.em = em;
}
@Override
protected Object getTargetRepository(RepositoryMetadata metadata) {
return new BaseRepositoryImpl((Class) metadata.getDomainType(), em);
}
@Override
protected Class> getRepositoryBaseClass(RepositoryMetadata metadata) {
return BaseRepositoryImpl.class;
}
}
}
配置Jpa factory class
最后,我们需要配置Jpa使用我们自定义的BaseRepositoryFactoryBean
。Spring支持使用标注进行配置,我们在com.tmy.App
中添加标注@EnableJpaRepositories(repositoryFactoryBeanClass = BaseRepositoryFactoryBean.class)
:
@SpringBootApplication
@EnableJpaRepositories(repositoryFactoryBeanClass = BaseRepositoryFactoryBean.class)
public class App {
public static void main( String[] args ){
SpringApplication.run(App.class, args);
}
}
这样我们就为所有Repository添加了自定义的方法。
测试
我们添加了一个TestController
进行测试。进入根目录,执行mvn spring-boot:run
可以运行我们的应用。
应用启动后,分别访问[[http://localhost:8080/test?type=blog&id=1](http://localhost:8080/test?type=blog&id=1)]([http://localhost:8080/test?type=blog&id=1](http://localhost:8080/test?type=blog&id=1))
和[[http://localhost:8080/test?type=article&id=1](http://localhost:8080/test?type=article&id=1)]([http://localhost:8080/test?type=article&id=1](http://localhost:8080/test?type=article&id=1))
,我们将看到article和blog两个不同的对象。
在TestController
中,我们通过依赖式注入获取到所有Repository的列表。当用户访问/test
,系统将根据传进来的type
遍历所有Repository,找到对应的Repository,再调用findOne(id)
方法找到对应的对象。这样我们就不需要一个一个的去获取Repository实例,当领域对象越来越多时,通过这种方式是一种更加高效的对象管理方法。
@RestController
public class TestController {
@Autowired
private List repositories;
@RequestMapping(value = "/test", method=RequestMethod.GET)
public Object getEntry(@RequestParam(value="type", required = true) String type,
@RequestParam(value="id", required=true) Integer id){
if(type.equals("article")){
type = Article.class.getName();
}else if (type.equals("blog")) {
type = Blog.class.getName();
}
for (BaseRepository baseRepository : repositories) {
if(baseRepository.support(type)){
return baseRepository.findOne(id);
}
}
return null;
}
}
版权声明
本文由Cliff创作,转载需署名作者且注明文章出处
参考代码
要获取本文的参考代码,请访问: http://tianmaying.com/tutorial/spring-jpa-custom-all/repo