【发布时间】:2015-03-13 01:37:42
【问题描述】:
我只是盯着spring-data 和spring-data-rest,我真的很想利用这些工具所提供的优势。在大多数情况下,基本功能非常适合我的用例,但是在某些情况下,我需要对底层功能进行大量自定义,并有选择地分配一些存储库来继承我所追求的自定义功能。
为了更好地解释这个问题,在spring-data 中有两个可能的接口,您可以从中继承功能,CrudRepository 或PagingAndSortingRepository。我想添加第三个称为PesimisticRepository
所有PesimisticRepository 所做的只是以不同方式处理已删除@Entity 的概念。 deleted 实体的deleted 属性为NOT NULL。这意味着可以由PesimisticRepository 处理的@Entity 必须具有 deleted 属性。
这一切都是可能的,我实际上在几年前就已经实现了。 (有兴趣的可以查看here)
我目前使用 spring-data 的尝试如下:
PagingAndSortingRepository 的扩展
package com.existanze.xxx.datastore.repositories;
import org.springframework.data.repository.NoRepositoryBean;
import org.springframework.data.repository.PagingAndSortingRepository;
import java.io.Serializable;
@NoRepositoryBean
public interface PesimisticRepository<T,ID extends Serializable> extends PagingAndSortingRepository<T,ID> {
}
为此我提供了一个扩展JPARepository的默认实现
package com.existanze.xxx.datastore.repositories;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.PageImpl;
import org.springframework.data.domain.Pageable;
import org.springframework.data.jpa.domain.Specification;
import org.springframework.data.jpa.repository.support.SimpleJpaRepository;
import org.springframework.transaction.annotation.Transactional;
import javax.persistence.EntityManager;
import javax.persistence.TypedQuery;
import javax.persistence.criteria.CriteriaBuilder;
import javax.persistence.criteria.CriteriaQuery;
import javax.persistence.criteria.Predicate;
import javax.persistence.criteria.Root;
import java.io.Serializable;
import java.util.Date;
public class JpaPesimisticRepository<T,ID extends Serializable> extends SimpleJpaRepository<T,ID> implements PesimisticRepository<T,ID> {
private final EntityManager entityManager;
public JpaPesimisticRepository(Class<T> domainClass, EntityManager em) {
super(domainClass, em);
this.entityManager = em;
}
@Override
@Transactional
public Page<T> findAll(Specification<T> spec, Pageable pageable) {
CriteriaBuilder cb = this.entityManager.getCriteriaBuilder();
CriteriaQuery<T> criteriaQuery = cb.createQuery(getDomainClass());
Root<T> from = criteriaQuery.from(this.getDomainClass());
Predicate deleted = cb.equal(from.get("deleted"), cb.nullLiteral(Date.class));
criteriaQuery.select(from).where(deleted);
TypedQuery<T> query = this.entityManager.createQuery(criteriaQuery);
return pageable == null ? new PageImpl<T>(query.getResultList()) : readPage(query, pageable, spec);
}
}
然后对于我希望使用悲观方法处理删除的任何 bean,我将其定义为这样
package com.existanze.xxx.datastore.repositories;
import com.existanze.xxx.domain.Phone;
import org.springframework.data.rest.core.annotation.RepositoryRestResource;
@RepositoryRestResource
public interface PhoneRepository extends PesimisticRepository<Phone,Integer> {
}
重要的是要解释为什么我希望覆盖这些方法而不是提供自定义方法,例如 findAllButDeleted。原因是因为我还希望悲观的删除能够渗透到spring-data-rest。这样生成的 HTTP 端点就不需要任何形式的自定义。
这似乎只适用于findAll 方法。但是对于其余的方法,当前的异常被抛出。
$ curl http://localhost:8881/phones/23
<html>
<head>
<meta http-equiv="Content-Type" content="text/html;charset=ISO-8859-1"/>
<title>Error 500 </title>
</head>
<body>
<h2>HTTP ERROR: 500</h2>
<p>Problem accessing /phones/23. Reason:
<pre> org.springframework.web.util.NestedServletException: Request processing failed; nested exception is org.springframework.dao.InvalidDataAccessApiUsageException: object is not an instance of declaring class; nested exception is java.lang.IllegalArgumentException: object is not an instance of declaring class</pre></p>
<hr /><i><small>Powered by Jetty://</small></i>
</body>
</html>
此外,我已经阅读了允许您更改所有存储库的默认 JpaRepository 的文档,但我需要在每个存储库的基础上执行此操作。
我希望我已经足够描述了。如果有什么需要更好的解释,请在 cmets 部分告诉我。
【问题讨论】:
标签: java spring spring-data spring-data-jpa spring-data-rest