2015-08-28 45 views
6

我有以下CDI托管bean:UnserializableDependencyException:WELD-001413:该Bean声明钝化范围,但具有非钝化能力的依赖

@Named 
@SessionScoped 
public class InfoPageController implements Serializable { 

    @Inject 
    private InfoPageMapper mapper; 

} 

它部署到GlassFish 4.1时抛出以下异常:

Exception while loading the app : CDI deployment failure:WELD-001413: The bean Managed Bean [class de.triaconsulting.cashyourgame.fe.controller.InfoPageController] with qualifiers [@Default @Any @Named] declares a passivating scope but has a non-passivation-capable dependency Managed Bean [class de.triaconsulting.cashyourgame.fe.mapper.InfoPageMapper] with qualifiers [@Any @Default] 
org.jboss.weld.exceptions.UnserializableDependencyException: WELD-001413: The bean Managed Bean [class de.triaconsulting.cashyourgame.fe.controller.InfoPageController] with qualifiers [@Default @Any @Named] declares a passivating scope but has a non-passivation-capable dependency Managed Bean [class de.triaconsulting.cashyourgame.fe.mapper.InfoPageMapper] with qualifiers [@Any @Default] 
    at org.jboss.weld.bootstrap.Validator.validateInjectionPointPassivationCapable(Validator.java:477) 
    at org.jboss.weld.bootstrap.Validator.validateInjectionPointForDeploymentProblems(Validator.java:395) 
    at org.jboss.weld.bootstrap.Validator.validateInjectionPoint(Validator.java:291) 
    at org.jboss.weld.bootstrap.Validator.validateGeneralBean(Validator.java:134) 
    at org.jboss.weld.bootstrap.Validator.validateRIBean(Validator.java:165) 
    at org.jboss.weld.bootstrap.Validator.validateBean(Validator.java:529) 
    at org.jboss.weld.bootstrap.Validator.validateBeans(Validator.java:515) 
    at org.jboss.weld.bootstrap.Validator.validateDeployment(Validator.java:490) 
    at org.jboss.weld.bootstrap.WeldStartup.validateBeans(WeldStartup.java:419) 
    at org.jboss.weld.bootstrap.WeldBootstrap.validateBeans(WeldBootstrap.java:90) 
    at org.glassfish.weld.WeldDeployer.event(WeldDeployer.java:225) 
    at org.glassfish.kernel.event.EventsImpl.send(EventsImpl.java:131) 
    at org.glassfish.internal.data.ApplicationInfo.load(ApplicationInfo.java:328) 
    at com.sun.enterprise.v3.server.ApplicationLifecycle.deploy(ApplicationLifecycle.java:496) 
    at com.sun.enterprise.v3.server.ApplicationLifecycle.deploy(ApplicationLifecycle.java:219) 
    at org.glassfish.deployment.admin.DeployCommand.execute(DeployCommand.java:491) 
    at com.sun.enterprise.v3.admin.CommandRunnerImpl$2$1.run(CommandRunnerImpl.java:539) 
    at com.sun.enterprise.v3.admin.CommandRunnerImpl$2$1.run(CommandRunnerImpl.java:535) 
    at java.security.AccessController.doPrivileged(Native Method) 
    at javax.security.auth.Subject.doAs(Subject.java:356) 
    at com.sun.enterprise.v3.admin.CommandRunnerImpl$2.execute(CommandRunnerImpl.java:534) 
    at com.sun.enterprise.v3.admin.CommandRunnerImpl$3.run(CommandRunnerImpl.java:565) 
    at com.sun.enterprise.v3.admin.CommandRunnerImpl$3.run(CommandRunnerImpl.java:557) 
    at java.security.AccessController.doPrivileged(Native Method) 
    at javax.security.auth.Subject.doAs(Subject.java:356) 
    at com.sun.enterprise.v3.admin.CommandRunnerImpl.doCommand(CommandRunnerImpl.java:556) 
    at com.sun.enterprise.v3.admin.CommandRunnerImpl.doCommand(CommandRunnerImpl.java:1464) 
    at com.sun.enterprise.v3.admin.CommandRunnerImpl.access$1300(CommandRunnerImpl.java:109) 
    at com.sun.enterprise.v3.admin.CommandRunnerImpl$ExecutionContext.execute(CommandRunnerImpl.java:1846) 
    at com.sun.enterprise.v3.admin.CommandRunnerImpl$ExecutionContext.execute(CommandRunnerImpl.java:1722) 
    at com.sun.enterprise.v3.admin.AdminAdapter.doCommand(AdminAdapter.java:534) 
    at com.sun.enterprise.v3.admin.AdminAdapter.onMissingResource(AdminAdapter.java:224) 
    at org.glassfish.grizzly.http.server.StaticHttpHandlerBase.service(StaticHttpHandlerBase.java:189) 
    at com.sun.enterprise.v3.services.impl.ContainerMapper$HttpHandlerCallable.call(ContainerMapper.java:459) 
    at com.sun.enterprise.v3.services.impl.ContainerMapper.service(ContainerMapper.java:167) 
    at org.glassfish.grizzly.http.server.HttpHandler.runService(HttpHandler.java:201) 
    at org.glassfish.grizzly.http.server.HttpHandler.doHandle(HttpHandler.java:175) 
    at org.glassfish.grizzly.http.server.HttpServerFilter.handleRead(HttpServerFilter.java:235) 
    at org.glassfish.grizzly.filterchain.ExecutorResolver$9.execute(ExecutorResolver.java:119) 
    at org.glassfish.grizzly.filterchain.DefaultFilterChain.executeFilter(DefaultFilterChain.java:284) 
    at org.glassfish.grizzly.filterchain.DefaultFilterChain.executeChainPart(DefaultFilterChain.java:201) 
    at org.glassfish.grizzly.filterchain.DefaultFilterChain.execute(DefaultFilterChain.java:133) 
    at org.glassfish.grizzly.filterchain.DefaultFilterChain.process(DefaultFilterChain.java:112) 
    at org.glassfish.grizzly.ProcessorExecutor.execute(ProcessorExecutor.java:77) 
    at org.glassfish.grizzly.nio.transport.TCPNIOTransport.fireIOEvent(TCPNIOTransport.java:561) 
    at org.glassfish.grizzly.strategies.AbstractIOStrategy.fireIOEvent(AbstractIOStrategy.java:112) 
    at org.glassfish.grizzly.strategies.WorkerThreadIOStrategy.run0(WorkerThreadIOStrategy.java:117) 
    at org.glassfish.grizzly.strategies.WorkerThreadIOStrategy.access$100(WorkerThreadIOStrategy.java:56) 
    at org.glassfish.grizzly.strategies.WorkerThreadIOStrategy$WorkerThreadRunnable.run(WorkerThreadIOStrategy.java:137) 
    at org.glassfish.grizzly.threadpool.AbstractThreadPool$Worker.doWork(AbstractThreadPool.java:565) 
    at org.glassfish.grizzly.threadpool.AbstractThreadPool$Worker.run(AbstractThreadPool.java:545) 
    at java.lang.Thread.run(Thread.java:745) 

这是如何引起的,我该如何解决?

+1

该消息提到'InfoPageMapper',但我没有在您的文章中看到任何定义 - 除了InfoPageController中的引用。消息被剥离了一下:'Bean Managed Bean [... InfoPageController] ...声明了一个钝化范围,但具有非钝化能力的依赖关系Managed Bean [... InfoPageMapper]' – Thomas

+1

我确实相信会话作用域bean的所有注入bean /属性都是可序列化的。它似乎是指向InfoPageMapper的手指。 – Gimby

回答

10

解决方案第一,其次是一个解释:

,你可以做最简单的就是纪念场InfoPageMapper mapperInfoPageControllertransient

@Named 
@SessionScoped 
public class InfoPageController implements Serializable { 

    @Inject 
    transient private InfoPageMapper mapper; 

而现在的解释:

错误消息以可读的语言陈述:

BeanSourcePageController,它是SessionScoped,必须是 可序列化的,但它需要InfoPageMapper,它不是 可序列化也不是瞬态的 - 不可能确定如何 序列化InfoPageController。

在CDI,还有一些范围(通常SessionScope),需要的bean被序列化 - 主要是因为他们已经在某种程度上连接到HTTP会话,它可以包含装入内存,并时不时的多个对象服务器可能需要将它们交换到磁盘。

看来你得到了这个,因为InfoPageController实现了Serializable。但根据Java系列化原则,这还不够。您需要确保Serializable类的所有成员字段都是以下值之一: - 原始类型(int,布尔型) - 可序列化的对象(所有序列化规则递归应用) - 该字段用关键字transient放置在相同的级别,如私人关键字)

CDI的诀窍是,您可以标记所有注入的字段为瞬态,因为当对象从磁盘反序列化到内存时它们再次注入。因此,当反序列化时,您不会丢失暂时对象,因为它以前没有存储到磁盘。

另一种解决方案是使注入的bean InfoPageMapper也可序列化。但随后问题可能会重复递归地注入到InfoPageMapper中的字段。瞬态关键字解决你的问题发生的地方,并不强制其他贝纳斯如果不需要是可序列化的。

+0

在标记字段瞬态之前,我将仔细检查字段是否属于会话应对bean。如果InfoPageMapper是一个具有默认范围(@Dependent)的bean,则隐式地将其设为会话范围的bean。所以它在每个会话中都被复制。如果映射器具有更多的控制器角色,则最好在其中注入会话作用域元素,每次调用时都会检索到正确的数据。 – Kazaag

+0

@Kazaag - 你是对的,在某些情况下,将注入的bean作为SessionScoped是有意义的。从技术角度来看,它也是一个Serializable bean,因此它可以解决异常问题。从逻辑角度来看,应用程序的行为可能会有所不同,因为多个bean将在会话范围内共享相同的bean。 – OndrejM

+0

也许有一点建议:对于我的测试用例,我也忘了将发现模式从beans中的“annotated”设置为“all”。 xml << version =“2.0”bean-discovery-mode =“all”>>。 – Bluefire

0

通过您的服务器错误消息,请检查de.triaconsulting.cashyourgame.fe.mapper.InfoPageMapper。 InfoPageMapper是否实现了Serializable?

您在InfoPageController对象的会话范围中注入InfoPageMapper对象。 会话范围对象及其成员需要序列化。所以InfoPageMapper对象必须能够序列化。

所以InfoPageMapper必须实现Serializable接口。

如果您不想在InfoPageController中序列化InfoPageMapper对象,则可以在此变量中设置transient关键字。

相关问题