gpt4 book ai didi

java.lang.ClassCastException : $Proxy96 cannot be cast to ticket. app.DatesFacade 错误

转载 作者:行者123 更新时间:2023-11-29 08:14:51 25 4
gpt4 key购买 nike

我正在将一些代码输入到在 Ubuntu 机器上运行的 Hudson 服务器中,以便在代码上运行一些 Code Metrics(Sonar)和 Cobertura。该项目基于 Glassfish3.1,使用 Maven3,用 Java、JSF 2.0 编写,并使用 OracleXE 数据库(无关紧要)。

尝试在我的 JUnit 测试中创建 Facade 实例时抛出错误。当我从 Netbeans 运行测试时,测试运行良好,但是当 Hudson 在其上进行自动构建时,我收到此错误:

    java.lang.ClassCastException: $Proxy96 cannot be cast to ticket.app.DatesFacade
at ticket.app.EventsControllerTest.setUp(EventsControllerTest.java:60)
at junit.framework.TestCase.runBare(TestCase.java:128)
at junit.framework.TestResult$1.protect(TestResult.java:106)
at junit.framework.TestResult.runProtected(TestResult.java:124)
at junit.framework.TestResult.run(TestResult.java:109)
at junit.framework.TestCase.run(TestCase.java:120)
at junit.framework.TestSuite.runTest(TestSuite.java:230)
at junit.framework.TestSuite.run(TestSuite.java:225)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:616)
at org.apache.maven.surefire.junit.JUnitTestSet.execute(JUnitTestSet.java:207)
at org.apache.maven.surefire.junit.JUnit3Provider.executeTestSet(JUnit3Provider.java:107)
at org.apache.maven.surefire.junit.JUnit3Provider.invoke(JUnit3Provider.java:79)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:616)
at org.apache.maven.surefire.booter.ProviderFactory$ClassLoaderProxy.invoke(ProviderFactory.java:103)
at $Proxy0.invoke(Unknown Source)
at org.apache.maven.surefire.booter.SurefireStarter.invokeProvider(SurefireStarter.java:145)
at org.apache.maven.surefire.booter.SurefireStarter.runSuitesInProcess(SurefireStarter.java:87)
at org.apache.maven.surefire.booter.ForkedBooter.main(ForkedBooter.java:69)

Hudson 服务器不是基于 JBoss,而是基于 Tomcat。我添加这个细节是因为当我用谷歌搜索这个错误时,我除了 JBoss 错误之外什么也没有发现,它解释了在 WAR 和 EAR 文件中都引用了外观。

编辑:这就是我目前创建 Facade 实例的方式(通过 Netbeans GF3.1 运行测试时效果很好):

        Map properties = new HashMap();
properties.put(EJBContainer.MODULES, new File("target/classes"));
properties.put("org.glassfish.ejb.embedded.glassfish.configuration.file", "server/config/domain.xml");
properties.put("oracle.jdbc.OracleDriver", "server/lib/ojdbc14.jar");
ejbContainer = EJBContainer.createEJBContainer(properties);
ctx = ejbContainer.getContext();

EventsFacade instance = (EventsFacade)ctx.lookup("java:global/classes/EventsFacade");

编辑:我的 EventsFacade:

@Stateful
public class EventsFacade extends AbstractFacade<Events> {
@PersistenceContext(unitName = "tickets_AppTicket_war_1.0-SNAPSHOTPU")
private EntityManager em;

protected EntityManager getEntityManager() {
return em;
}

public EventsFacade() {
super(Events.class);
}
}

编辑:还有..我完整的AbstractFacade:

public abstract class AbstractFacade<T> {
private Class<T> entityClass;

public AbstractFacade(Class<T> entityClass) {
this.entityClass = entityClass;
}

protected abstract EntityManager getEntityManager();

public void create(T entity) {
getEntityManager().persist(entity);
}

public void edit(T entity) {
getEntityManager().merge(entity);
}

public void remove(T entity) {
getEntityManager().remove(getEntityManager().merge(entity));
}

public T find(Object id) {
return getEntityManager().find(entityClass, id);
}

public List<T> findAll() {
javax.persistence.criteria.CriteriaQuery cq = getEntityManager().getCriteriaBuilder().createQuery();
cq.select(cq.from(entityClass));
return getEntityManager().createQuery(cq).getResultList();
}

public List<T> findRange(int[] range) {
javax.persistence.criteria.CriteriaQuery cq = getEntityManager().getCriteriaBuilder().createQuery();
cq.select(cq.from(entityClass));
javax.persistence.Query q = getEntityManager().createQuery(cq);
q.setMaxResults(range[1] - range[0]);
q.setFirstResult(range[0]);
return q.getResultList();
}

public int count() {
javax.persistence.criteria.CriteriaQuery cq = getEntityManager().getCriteriaBuilder().createQuery();
javax.persistence.criteria.Root<T> rt = cq.from(entityClass);
cq.select(getEntityManager().getCriteriaBuilder().count(rt));
javax.persistence.Query q = getEntityManager().createQuery(cq);
return ((Long) q.getSingleResult()).intValue();
}

编辑: 创建新界面:

    @Local
public interface EventsInterface<Events> {

int count();

void create(Events entity);

void edit(Events entity);

Events find(Object id);

List<Events> findAll();

List<Events> findRange(int[] range);

void remove(Events entity);

}

更改: EventsFacade 声明:

public class EventsFacade extends AbstractFacade<Events> implements EventsInterface<Events> {

编辑:在创建新界面后我的 IDE 中出现代理错误后,输出开始显示:

Field Name == intfClass
Field == interface ticket.app.EventsInterface
Field Name == containerId
Field == 85313541807800321
Field Name == delegate
Field == com.sun.ejb.containers.EJBLocalObjectInvocationHandler@72b0f2b2
Field Name == isOptionalLocalBusinessView
Field == false

最佳答案

这通常意味着当对象周围有代理(并且代理基于接口(interface))时,您通过类引用对象,而不是通过接口(interface)引用对象。

解决方案是通过接口(interface)引用对象。我猜 EventsFacade 有一个 Local 和/或 Remote 接口(interface)。尝试使用它。如果它没有界面 - 制作一个,这是一个很好的做法。在接口(interface)中定义所有公共(public)方法。

更新:看起来,这可能是您使用的嵌入式 glassfish 的问题。由于您无法使用调试器对其进行调试,因此您可以执行以下操作来跟踪问题:使用 java.lang.reflect.Proxy.getInvocationHandler(object)。然后列出返回对象的所有字段。 (您现在必须放弃强制转换并从上下文中简单地将其作为 Object 获取)。像这样的东西:

Object ic = Proxy.getInvocationHandler(facade);
Field[] fields = ic.getClass().getDeclaredFields();
for (Field field : fields) {
field.setAccessible(true);
System.out.println(field.getName() + "=" + field.get(ic);
}

关于java.lang.ClassCastException : $Proxy96 cannot be cast to ticket. app.DatesFacade 错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5524646/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com