3

我的代码中有一个非常奇怪的错误。我从项目开始就使用迭代器,我从来没有遇到过问题,但在这里我不明白发生了什么。

我有一个模型课

public class MyObject implements Serializable{
    private static final long serialVersionUID = 1L;

    private int field1;
    private String field2;
    private List<OtherObject> field3;
    private Date field4
    ...
    // + Getters and setters
    // + Override equals method
}

具有 MyObject 类型参数的类

public class FooClass implements Serializable{
    private static final long serialVersionUID = 1L;
    private List<MyObject> list;
    // + getters and setters
}

我还有一个使用 MyObject 的类

public class MyController {
    .....
    public static void AMethod(FooClass value) {
        List<MyObject> myList = value.getList();
        Iterator<MyObject> iterator = myList.iterator();
        while(iterator.hasNext()) {
            MyObject temp = iterator.next();  // error on this line
        }
    }
    ...
}

我的对象使用以下代码显示在 JSF 视图中:

<p:selectCheckboxMenu value="#{fooClass.list}">
    <f:selectItems value="#{fooClass.listAll}" var="obj" 
                   itemValue="#{obj}" itemLabel="#{obj.field2}" />
</p:selectCheckboxMenu>

这是我得到的错误:

java.lang.ClassCastException: java.lang.String cannot be cast to com.MyObject

有什么建议吗?

4

1 回答 1

2

发生这种情况是因为 HTTP 协议的“设计”:当请求发送到服务器时,所有数据都以Strings 发送。因此,JSFString也将它们解释为 s ,因为在您的情况下,您没有明确告诉预期的对象属于您的特定类MyObject。发生这种情况是因为处理所有视图模型通信的 EL 是基于反射的,并且您知道 Java 中的泛型由于类型擦除是一种编译时现象:泛型信息在运行时不可用。

因此,基本上在您提交表单后,您的列表由普通String对象组成,而不是您的MyObject实例,正如您似乎期望的那样:在提交表单后,通用信息被 JSF EL 替换。这就是为什么你得到ClassCastException. 如果在 action(listener) 方法或 setter 方法上放置断点,您可以自行检查列表包含的元素类型。

要解决这种情况,您需要明确告诉 JSF 使用 a Converter(通过指定converter属性或嵌套<f:converter>标签),或者通过切换到普通数组 (to MyObject[]) 而不是 a List<MyObject>

您可以在我对JSF 和类型安全问题的回答中找到更多信息以及一些解决方案。

于 2013-07-22T12:46:22.487 回答