Padrão DAO abstrato e o problema “Proxy não pode ser convertido em…” do Spring!
Sei que isso é muito solicitado, mas não consigo encontrar uma solução de trabalho:
Este é o meu AbstractDAO:
public interface AbstractDao<T>
{
public T get(Serializable id);
//other CRUD operations
}
E esta é a implementação do meu JPA:
public abstract class AbstractDaoJpaImpl<T> implements AbstractDao<T> , Serializable
{
protected EntityManager em;
protected Class<T> clazz;
@SuppressWarnings("unchecked")
public AbstractDaoJpaImpl()
{
ParameterizedType genericSuperclass = (ParameterizedType) getClass().getGenericSuperclass();
this.clazz = (Class<T>) genericSuperclass.getActualTypeArguments()[0];
}
public abstract void setEntityManager(EntityManager em);
//implementations skipped
}
E este é o dao de uma entidade:
public interface PersonDao extends AbstractDao<Person>
{
//empty
}
Aqui está a sua implementação:
@Repository
public class PersonDaoImpl extends AbstractDaoJpaImpl<Person> implements PersonDao , OtherInterface
{
@PersistenceContext(unitName="company")
@Override
public void setEntityManager(EntityManager em)
{
this.em = em;
}
@Override // implements OtherInterface.additionalMethods()
public additionalMethods()
{
// implements...
}
}
Toda a arquitetura é simples:
InterfaceAbstractDao define métodos simples de CRUD.
InterfacePersonDao estende AbstractDAO sem nenhum método adicional.
classeAbstractDaoJpaImpl define a implementação do AbstractDao pela JPA
classePersonDaoImpl estende AbstractDaoJpaImpl e implementa PersonDaoAND OtherInterface, que adiciona aditionalMethods ()...
SE, PersonDaoImpl implementa apenas PersonDao, sem implementar OtherInterface.additionalMethods (), tudo funciona bem.
Eu posso usar
<tx:annotation-driven transaction-manager="transactionManager" />
no arquivo XML da minha primavera.
MAS, PersonDaoImpl implementa OtherInterface (s), ao testar / executar, Iprecisa transmitir o DAO de PersonDao para PersonDaoImpl ou OtherInterfaces , tal como :
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations={"classpath:app.xml"})
@TransactionConfiguration(transactionManager="transactionManager" , defaultRollback=false)
public class PersonDaoTest
{
@Inject
PersonDao dao;
@Test
public void testAdditionalMethod()
{
PersonDaoImpl impl = (PersonDaoImpl) dao;
System.out.println(impl.additionalMethod(...));
}
}
O problema ocorre quando(PersonDaoImpl) dao
, que lança a exceção "O proxy não pode ser convertido para PersonDaoImpl":
java.lang.ClassCastException: $Proxy36 cannot be cast to foobar.PersonDaoImpl
at foobar.PersonDaoTest.testAdditionalMethod(PersonDaoTest.java:36)
isso é frequentemente solicitado no google, todos sugerem adicionarproxy-target-class="true"
para<tx:annotation-driven>
:
<tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true" />
Isso fará uso do CGLIB em vez do proxy dinâmico do JDK.
MAS lança outra exceção ao inicializar o Spring:
Caused by: java.lang.ClassCastException: java.lang.Class cannot be cast to java.lang.reflect.ParameterizedType
no construtor de AbstractDaoJpaImpl:
ParameterizedType genericSuperclass = (ParameterizedType) getClass().getGenericSuperclass();
Toda pergunta pára por aqui, não consigo encontrar nenhumatrabalhando soluções agora.
Alguém pode me dar uma solução de trabalho? Muito obrigado !
Ambiente: Spring-3.0.4, javaee-api-6.0, javax.inject, cglib-2.2, hibernate-jpa-2.0-api-1.0.0,