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,

questionAnswers(2)

yourAnswerToTheQuestion