UpdateModelException ClassCastException в JSF 2.0 - PullRequest
2 голосов
/ 23 октября 2011

У меня есть страница facelets с тегом ui: repeat, который выводит строки таблицы HTML, и каждая строка принимает пользовательский ввод. Это работает штрафы. У меня также есть второй набор данных, который я выводю, используя ту же страницу и повторитель (просто разные данные). Основным компонентом, поддерживающим эту страницу, является ConversationScoped.

<ui:repeat value="#{cloneBuilderBean.pageTemplate}" var="row" varStatus="status">
  <tr>
    <td>#{row.label}</td>
    <td><h:inputText value=#{row.value}/></td>
  </tr>
</ui:repeat>

Проблема, с которой я столкнулся, заключается в том, что пользователь нажимает кнопку h: commandButton, чтобы «перейти» на следующую страницу. Это действительно просто отображает текущую страницу, но коллекция повторителя была обновлена ​​до нового набора данных. Из моей отладки не похоже, что я когда-либо даже достиг своего метода действия, который должен сработать в результате нажатия кнопки commandButton для перехода на следующую страницу. Трассировка стека ниже также не показывает никаких признаков попадания в мой код. Любые мысли о том, как решить эту ошибку?

javax.faces.component.UpdateModelException: java.lang.ClassCastException
    at javax.faces.component.UIInput.updateModel(UIInput.java:853)
    at javax.faces.component.UIInput.processUpdates(UIInput.java:735)
    at com.sun.faces.facelets.component.UIRepeat.process(UIRepeat.java:525)
    at com.sun.faces.facelets.component.UIRepeat.processUpdates(UIRepeat.java:748)
    at javax.faces.component.UIForm.processUpdates(UIForm.java:281)
    at javax.faces.component.UIComponentBase.processUpdates(UIComponentBase.java:1242)
    at javax.faces.component.UIComponentBase.processUpdates(UIComponentBase.java:1242)
    at javax.faces.component.UIViewRoot.processUpdates(UIViewRoot.java:1231)
    at com.sun.faces.lifecycle.UpdateModelValuesPhase.execute(UpdateModelValuesPhase.java:78)
    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 org.apache.catalina.core.StandardWrapper.service(StandardWrapper.java:1539)
    at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:343)
    at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:217)
    at org.primefaces.webapp.filter.FileUploadFilter.doFilter(FileUploadFilter.java:79)
    at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:256)
    at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:217)
    at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:279)
    at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:175)
    at org.apache.catalina.core.StandardPipeline.doInvoke(StandardPipeline.java:655)
    at org.apache.catalina.core.StandardPipeline.invoke(StandardPipeline.java:595)
    at com.sun.enterprise.web.WebPipeline.invoke(WebPipeline.java:98)
    at com.sun.enterprise.web.PESessionLockingStandardPipeline.invoke(PESessionLockingStandardPipeline.java:91)
    at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:162)
    at org.apache.catalina.connector.CoyoteAdapter.doService(CoyoteAdapter.java:330)
    at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:231)
    at com.sun.enterprise.v3.services.impl.ContainerMapper.service(ContainerMapper.java:174)
    at com.sun.grizzly.http.ProcessorTask.invokeAdapter(ProcessorTask.java:828)
    at com.sun.grizzly.http.ProcessorTask.doProcess(ProcessorTask.java:725)
    at com.sun.grizzly.http.ProcessorTask.process(ProcessorTask.java:1019)
    at com.sun.grizzly.http.DefaultProtocolFilter.execute(DefaultProtocolFilter.java:225)
    at com.sun.grizzly.DefaultProtocolChain.executeProtocolFilter(DefaultProtocolChain.java:137)
    at com.sun.grizzly.DefaultProtocolChain.execute(DefaultProtocolChain.java:104)
    at com.sun.grizzly.DefaultProtocolChain.execute(DefaultProtocolChain.java:90)
    at com.sun.grizzly.http.HttpProtocolChain.execute(HttpProtocolChain.java:79)
    at com.sun.grizzly.ProtocolChainContextTask.doCall(ProtocolChainContextTask.java:54)
    at com.sun.grizzly.SelectionKeyContextTask.call(SelectionKeyContextTask.java:59)
    at com.sun.grizzly.ContextTask.run(ContextTask.java:71)
    at com.sun.grizzly.util.AbstractThreadPool$Worker.doWork(AbstractThreadPool.java:532)
    at com.sun.grizzly.util.AbstractThreadPool$Worker.run(AbstractThreadPool.java:513)
    at java.lang.Thread.run(Thread.java:662)
Caused by: java.lang.ClassCastException
    at javax.el.ArrayELResolver.setValue(ArrayELResolver.java:257)
    at com.sun.faces.el.DemuxCompositeELResolver._setValue(DemuxCompositeELResolver.java:255)
    at com.sun.faces.el.DemuxCompositeELResolver.setValue(DemuxCompositeELResolver.java:281)
    at com.sun.el.parser.AstValue.setValue(AstValue.java:197)
    at com.sun.el.ValueExpressionImpl.setValue(ValueExpressionImpl.java:286)
    at org.jboss.weld.el.WeldValueExpression.setValue(WeldValueExpression.java:74)
    at com.sun.faces.facelets.el.TagValueExpression.setValue(TagValueExpression.java:131)
    at javax.faces.component.UIInput.updateModel(UIInput.java:818)
    ... 41 more

Обновление : соответствующий код компонента:

private int[] part1,part2;
private int[] currentPart; //pointer to part1 or part2
private List<SelectItem> pageTemplate;

public String openItem(String value){
    conversation.begin();

    //database dummy data...
    pageTemplate = new ArrayList(4);
    pageTemplate.add(new SelectItem("Data1"));
    pageTemplate.add(new SelectItem("Data2"));
    pageTemplate.add(new SelectItem("Data3"));
    pageTemplate.add(new SelectItem("Data4"));

    //database dummy data...
    part1 = new int[4];
    for(int i = 0; i < part1.length; i++){
        part1[i] = i+1;
    }
    currentPart = part1;

    return "page2";
}

public String next(){
    //database dummy data...
    part2 = new int[4];
    for(int i = 0; i < part2.length; i++){
        part2[i] = i+2;
    }
    currentPart = part2;
    return "page2";
}

соответствующий код вида:

<ui:repeat value="#{CloneBuilderBean.pageTemplate}" var="row" varStatus="status">
    <tr>
        <td>#{row.value}</td>
        <td width="300">
            <h:inputHidden id="val" value="#{CloneBuilderBean.currentPart[status.index]}" />
            <p:slider for="val" step="1" style="width:90%;" minValue="0" maxValue="7" />
        </td>
    </tr>
</ui:repeat>

<h:commandButton value="display part 2" action="#{CloneBuilderBean.next()}"/>

1 Ответ

2 голосов
/ 23 октября 2011

Причина: java.lang.ClassCastException
at javax.el.ArrayELResolver.setValue (ArrayELResolver.java:257)

Я воспроизвел вашу проблему. Оказывается, #{CloneBuilderBean.currentPart[status.index]} было установлено с Integer вместо int. Это работает для меня, когда я заменяю int[] на Integer[].

Когда я попытался воспроизвести вашу проблему на Tomcat 7.0.19 вместо Glassfish 3.1.1, я получил более четкое исключение из его анализатора EL:

Причина: java.lang.ClassCastException: Невозможно добавить объект типа [java.lang.Integer] в массив объектов типа [int]
at javax.el.ArrayELResolver.setValue (ArrayELResolver.java:96)

...