5

挣扎了好几个小时,我终于找到了那些烦人ClassCastException的 s 是从哪里来的,我以为是 Hibernate 产生的,它是enum-mapping。

但它们来自我的 JSF 视图,我List从那里传递了一个

    <h:selectManyCheckbox value="#{createUserManager.user.roles}"  ... >
        <f:selectItems value="#{createUserManager.roles}"/>
    </h:selectManyCheckbox>

回到我的支持豆。
我的数据仅由枚举的值组成: public Role[] getRoles() { return Role.values(); }.
当我rolesUser-class 中测试 setter 并得到以下结果时,我真的很震惊:

public void setRoles(List<Role> paramRoles) {

    System.out.println(paramRoles.get(0) instanceof Role); //output: false

    for(Role role : paramRoles){ ...} //crashes with ClassCastException
}

更改List<Role> paramRolesList<String> paramRoles完美运行。
这怎么可能?这些泛型不应该是类型安全的,还是与 JSF 相关的类型擦除会扼杀整个类型安全?也不应该
是的返回值,就像我通过?h:selectManyCheckboxList<Role>f:selectItems

4

1 回答 1

7

您正在经历的行为是完全可以预料的。此外,它与 Java 泛型的关系与 HTTP 的工作方式相同。

问题

  1. HTTP 部分

    问题是您不完全了解 HTTP 的工作原理。当您通过按下提交按钮提交数据时,您由 JSF<h:selectManyCheckbox>标签生成的参数,作为一堆<input type="checkbox" name="..." value="userRoleAsString">复选框,将作为字符串发送并最终以request.getParameter("checkboxName");字符串形式检索。当然,JSF 不知道如何构造模型对象类,Role.

  2. 泛型部分

    如您所知,由于 java 为泛型选择类型擦除以提供向后兼容性,有关泛型类型的信息基本上是编译类型的工件,并且在运行时丢失。因此,在运行时,您将List<Role>擦除到一个普通的、好的旧的List. 就 EL 是一种使用 Java 反射 API 来处理您的表达式/调用方法的运行时语言而言,在运行时没有这样的信息可用。考虑到 HTTP 部分,JSF 尽其所能并将字符串对象分配给您的列表,因为这是它可以隐式执行的所有操作。如果您愿意告诉 JSF 不这样做,您需要明确地这样做,即通过指定一个转换器来知道在 HTTP 请求中期望什么类型的对象

  3. JSF 部分:后果

    JSF 提供了一个javax.faces.Enum转换器,如果 EL 知道您的列表的编译时泛型类型(即Role. 但它不知道。如果您对一个Role[] userRoles对象进行多项选择,或者如果您使用唯一选择(如 in<h:selectOneMenu>中的值绑定到Role userRole. 在这些示例中,将自动调用内置的枚举转换器。

    因此,要使其按预期工作,您需要提供一个Converter“解释”JSF 这个列表包含什么类型的值,以及如何进行从Roleto的转换String,反之亦然。

    值得注意的是,List<...>多选 JSF 组件中的任何绑定值都会出现这种情况。


Stack Overflow 上的参考点

在检查并解决问题后,我想知道过去是否没有人遇到过它并在这里搜索了一些以前的答案。不出意外,之前有人问过,问题当然是被BalusC解决了。以下是两个最有价值的参考点:


测试用例和工作转换器的两个示例

下面我提供一个测试用例完全供您理解:除了第三个<h:selectManyCheckbox>组件之外,一切都按预期工作。您可以完全跟踪它以完全消除问题。

风景:

<h:form>
    Many with enum converter
    <!-- will be mapped correctly with Role object -->
    <h:selectManyCheckbox value="#{q16433250Bean.userRoles}" converter="roleEnumConverter">
        <f:selectItems value="#{q16433250Bean.allRoles}" var="role" itemLabel="#{role.name}" />
    </h:selectManyCheckbox>
    <br/>
    Many with plain converter
    <!-- will be mapped correctly with Role object -->
    <h:selectManyCheckbox value="#{q16433250Bean.userRoles2}" converter="roleConverter">
        <f:selectItems value="#{q16433250Bean.allRoles2}" var="role" itemLabel="#{role.name}" />
    </h:selectManyCheckbox>
    <br/>
    Without any converter
    <!-- will NOT be mapped correctly with Role object, but with a default String instead -->
    <h:selectManyCheckbox value="#{q16433250Bean.userRoles3}">
        <f:selectItems value="#{q16433250Bean.allRoles}" var="role" itemLabel="#{role.name}" />
    </h:selectManyCheckbox>
    <br/>
    Without any converter + array
    <!-- will be mapped correctly with Role object -->
    <h:selectManyCheckbox value="#{q16433250Bean.userRoles4}">
        <f:selectItems value="#{q16433250Bean.allRoles}" var="role" itemLabel="#{role.name}" />
    </h:selectManyCheckbox>
    <br/>
    <h:commandButton value="Submit" action="#{q16433250Bean.action}"/>
</h:form>

豆子:

@ManagedBean
@RequestScoped
public class Q16433250Bean {

    private List<Role> userRoles = new ArrayList<Role>();//getter + setter
    private List<Role> userRoles2 = new ArrayList<Role>();//getter + setter
    private List<Role> userRoles3 = new ArrayList<Role>();//getter + setter
    private Role[] userRoles4;//getter + setter

    public enum Role {

        ADMIN("Admin"),
        SUPER_USER("Super user"),
        USER("User");
        private final String name;

        private Role(String name) {
            this.name = name;
        }

        public String getName() {
            return this.name;
        }
    }

    public Role[] getAllRoles() {
        return Role.values();
    }

    public String action() {
        return null;
    }

}

转换器:

@FacesConverter("roleEnumConverter")
public class RoleEnumConverter extends EnumConverter {

    public RoleEnumConverter() {
        super(Role.class);
    }

}

@FacesConverter("roleConverter")
public class RoleConverter implements Converter {

    public Object getAsObject(FacesContext context, UIComponent component, String value) {
        if(value == null || value.equals("")) {
            return null;
        }
        Role role = Role.valueOf(value);
        return role;
    }

    public String getAsString(FacesContext context, UIComponent component, Object value) {
        if (!(value instanceof Role) || (value == null)) {
            return null;
        }
        return ((Role)value).toString();
    }

}
于 2013-05-08T06:08:43.020 回答