【问题标题】:Primefaces datatable selected row not working using viewscopePrimefaces数据表所选行无法使用viewscope
【发布时间】:2012-12-26 13:12:48
【问题描述】:

我正在使用 JSF 2.0 和 Primefaces 3.4.2,我有一个使用延迟加载填充的数据表。

当我查看 managedbean 的范围时,datatable selectedRow 给出了空指针异常。 如果我使用会话范围,那么我可以在 managedbean 中获得 selectedRow。

我正在使用 CDI Spring 注释来指定范围。我使用this 方法来创建视图范围。

更新 1

我注意到另一件事是当我分页到第二页然后返回第一页时使用视图范围,然后我可以获得 selectedRow。如果我选择没有分页的行,则会出现空指针异常。

JSF 页面

<p:dataTable id="dataTable" var="req" lazy="true" value="#{emp.lazyModel}"
                paginator="true" rows="10" 
                             selection="#{emp.selectedRequest}"
                            selectionMode="single">                         
          <p:ajax event="rowSelect" listener="#{emp.onRowSelect}" />  

托管豆

@Named("emp")
@Scope("view")
public class EmployeesManagedBean implements Serializable {

    @PostConstruct
    public void init() {
        initTable();
    }

    private void initTable() {
        lazyModel = new LazyRequestDataModel(requestList, requestService);
    }

    public LazyDataModel<Employee> getLazyModel() {
        return lazyModel;

    }

我在 onRowSelect 方法的这一行得到空指针异常

 Emp emp = (Emp) event.getObject()); 
 System.out.println(emp.getEmpNo() );

完整的错误堆栈跟踪

java.lang.NullPointerException
    at net.test.managed.bean.RequestManagedBean.onRowSelect(RequestManagedBean.java:134)
    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:601)
    at com.sun.el.parser.AstValue.invoke(AstValue.java:187)
    at com.sun.el.MethodExpressionImpl.invoke(MethodExpressionImpl.java:297)
    at org.primefaces.component.behavior.ajax.AjaxBehaviorListenerImpl.processAjaxBehavior(AjaxBehaviorListenerImpl.java:52)
    at org.primefaces.event.SelectEvent.processListener(SelectEvent.java:40)
    at javax.faces.component.behavior.BehaviorBase.broadcast(BehaviorBase.java:106)
    at javax.faces.component.UIComponentBase.broadcast(UIComponentBase.java:760)
    at javax.faces.component.UIData.broadcast(UIData.java:1071)
    at javax.faces.component.UIData.broadcast(UIData.java:1093)
    at javax.faces.component.UIViewRoot.broadcastEvents(UIViewRoot.java:794)
    at javax.faces.component.UIViewRoot.processApplication(UIViewRoot.java:1259)
    at com.sun.faces.lifecycle.InvokeApplicationPhase.execute(InvokeApplicationPhase.java:81)
    at com.sun.faces.lifecycle.Phase.doPhase(Phase.java:101)
    at com.sun.faces.lifecycle.LifecycleImpl.execute(LifecycleImpl.java:118)
    at javax.faces.webapp.FacesServlet.service(FacesServlet.java:593)
    at weblogic.servlet.internal.StubSecurityHelper$ServletServiceAction.run(StubSecurityHelper.java:227)
    at weblogic.servlet.internal.StubSecurityHelper.invokeServlet(StubSecurityHelper.java:125)
    at weblogic.servlet.internal.ServletStubImpl.execute(ServletStubImpl.java:301)
    at weblogic.servlet.internal.TailFilter.doFilter(TailFilter.java:26)
    at weblogic.servlet.internal.FilterChainImpl.doFilter(FilterChainImpl.java:56)
    at weblogic.servlet.internal.RequestEventsFilter.doFilter(RequestEventsFilter.java:27)
    at weblogic.servlet.internal.FilterChainImpl.doFilter(FilterChainImpl.java:56)
    at weblogic.servlet.internal.WebAppServletContext$ServletInvocationAction.wrapRun(WebAppServletContext.java:3730)
    at weblogic.servlet.internal.WebAppServletContext$ServletInvocationAction.run(WebAppServletContext.java:3696)
    at weblogic.security.acl.internal.AuthenticatedSubject.doAs(AuthenticatedSubject.java:321)
    at weblogic.security.service.SecurityManager.runAs(SecurityManager.java:120)
    at weblogic.servlet.internal.WebAppServletContext.securedExecute(WebAppServletContext.java:2273)
    at weblogic.servlet.internal.WebAppServletContext.execute(WebAppServletContext.java:2179)
    at weblogic.servlet.internal.ServletRequestImpl.run(ServletRequestImpl.java:1490)
    at weblogic.work.ExecuteThread.execute(ExecuteThread.java:256)
    at weblogic.work.ExecuteThread.run(ExecuteThread.java:221)

【问题讨论】:

    标签: java jsf-2 primefaces nullpointerexception viewstate


    【解决方案1】:

    我已经设法通过使用 CDI 支持的@ViewAccessScoped 解决了这个问题。在maven中添加依赖或者直接从http://myfaces.apache.org/extensions/cdi/download.html下载,放到classpath中。

    <dependency>
        <groupId>org.apache.myfaces.extensions.cdi.core</groupId>
        <artifactId>myfaces-extcdi-core-api</artifactId>
        <version>1.0.5</version>
        <scope>compile</scope>
    </dependency>
    

    仍然存在一个小问题,即第一次部署应用程序时,我仍然没有获得选定的行值,对于后续的选择,我能够获得选定的行值。

    【讨论】:

    • 你应该继续调查这个问题,因为最终它应该从一开始就可以正常工作,而不是在第一次之后
    • @Daniel 我不知道它是如何第一次没有被选中的行。
    • 如何修改你的get()remove() 方法,如here 提到的那样在它们两个中添加额外的条件检查?
    • PS:这对我有用,但我没有像你那样使用 CDI bean。
    • @Tiny 基本上是 Spring 的视图范围而不是 CDI 对吧?
    【解决方案2】:

    如果你想使用选择,我认为你必须在数据表上提供 rowKey 属性。

    【讨论】:

    • 如果我添加 rowKey 那么我会得到异常 java.lang.NullPointerException at org.primefaces.component.datatable.DataTable.getRowData(DataTable.java:798) at org.primefaces.component.datatable.feature.SelectionFeature.decodeSingleSelection(SelectionFeature.java:46) 此外我不确定 PF 3.4.2 是否需要 rowKey,例如,请参阅 this
    • 这个例子有rowKey,我猜你想的是延迟加载的例子。该示例在标签上没有 rowKey,但他们将其放入 LazyDataModel 实现中。他们编写了 getRowKey 和 getRowData 方法。我对此有一个理论:您在页面上有一些 c: 标签,或者您有一些绑定属性。如果您提供完整的 xhtml 代码,将会很有帮助。我认为这是问题所在,因为您告诉过会话范围 bean 一切正常。
    猜你喜欢
    • 1970-01-01
    • 2014-01-23
    • 1970-01-01
    • 2015-05-22
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2013-12-28
    • 1970-01-01
    相关资源
    最近更新 更多