spring data jpa使用自定义repository实现类
spring data jpa中使用JpaRepository等接口定义repository时,将默认使用SimpleJpaRepository
可通过自定义实现类,改写或自定义接口方法逻辑:
创建MyJpaRepository实现类
- @Repository
- @Transactional(readOnly = true)
- public class MyRepositoryImpl<T,ID> extends SimpleJpaRepository<T, ID> {
- public MyRepositoryImpl(JpaEntityInformation<T, ID> entityInformation, EntityManager entityManager) {
- super(entityInformation, entityManager);
- }
- public MyRepositoryImpl(Class<T> domainClass, EntityManager em) {
- super(domainClass, em);
- }
- ...
- }
创建MyJpaRepositoryFactoryBean
- public class MyJpaRepositoryFactoryBean<R extends Repository<T, ID>, T, ID>
- extends JpaRepositoryFactoryBean<R, T, ID> {
- public MyJpaRepositoryFactoryBean(Class<? extends R> repositoryInterface) {
- super(repositoryInterface);
- }
- @SuppressWarnings("rawtypes")
- protected RepositoryFactorySupport createRepositoryFactory(EntityManager em) {
- return new MyJpaRepositoryFactory(em);
- }
- private static class MyJpaRepositoryFactory<T, ID>
- extends JpaRepositoryFactory {
- private final EntityManager em;
- public MyJpaRepositoryFactory(EntityManager em) {
- super(em);
- this.em = em;
- }
- @Override
- protected JpaRepositoryImplementation<?, ?> getTargetRepository(RepositoryInformation information, EntityManager entityManager) {
- return new EntityJPARepositoryImpl<E>(
- (Class<E>) information.getDomainType(), entityManager);
- }
- protected Class<?> getRepositoryBaseClass(RepositoryMetadata metadata) {
- return MyRepositoryImpl.class;
- }
- }
- }
配置JPA
- @Configuration
- @EnableJpaRepositories(basePackages = {...},
- repositoryFactoryBeanClass = MyJpaRepositoryFactoryBean.class)
- public class JPAConfig {}
Jpa自定义Repository方法
如果不使用SpringData的方法,想要自己实现,该怎么办呢?
定义一个接口: 声明要添加的, 并自实现的方法
提供该接口的实现类: 类名需在要声明的 Repository 后添加 Impl, 并实现方法
声明 Repository 接口, 并继承 1) 声明的接口
注意: 默认情况下, Spring Data 会在 base-package 中查找 "接口名Impl" 作为实现类. 也可以通过 repository-impl-postfix 声明后缀.
这张图是类与接口之间的关系

下面是具体的实现:
包结构

类与接口之间的关系代码
- public interface PersonRepositoiry extends JpaRepository<Person, Integer> ,PersonDao{
- public interface PersonDao {
- void test();
- }
- @Repository
- public class PersonRepositoiryImpl implements PersonDao{
- @PersistenceContext
- private EntityManager em;
- @Override
- public void test() {
- //只是用来测试
- Person person = em.find(Person.class, 1);
- System.out.println(person);
- }
- }
测试代码
- @Test
- public void testCustomerRepositoryMethod() {
- personRepositoiry.test();
- }
经过实践发现
- XXXRepositoryImpl 与XXXRepository前面的名字必须相同,后面的也需要按照规则写
- 若将XXXRepositoryImpl与XXXRepository接口放在同意包下,XXXRepositoryImpl不需要添加@Repository注解,但是当XXXRepositoryImpl与XXXRepository接口不在同一包下,需要在在XXXRepositoryImpl类上加@Repository注解进行修饰
以上为个人经验,希望能给大家一个参考,也希望大家多多支持w3xue。