为什么在春季我会得到IllegalArgumentException无法将String类型的值转换为所需类型Product的原因?


问题内容

我收到例外

无法将属性产品的类型[java.lang.String]的属性值转换为必需的类型[beans.Product];嵌套异常为java.lang.IllegalArgumentException:无法将[java.lang.String]类型的值转换为属性产品的必需类型[beans.Product]:找不到匹配的编辑器或转换策略

在Errors错误对象中,甚至在我的DetailProductValidator开始通过validate方法进行验证之前。

我不明白为什么Spring会这样做。我没有直接映射到产品属性/对象的任何输入字段。我只是在jsp中使用产品对象的属性。例如,我使用:

<form:options items="${dpBackObj.product.colorMap}"/>
<!-- or -->
${dpBackObj.product.priceInDollars}

但我从不使用:

<form:input path="product"/>

谁能解释为什么会这样? 也许告诉我一个简单的解决方案?

控制器的Bean配置为:

    <!-- DETAIL PRODUCT FORM CONTROLLER -->
<bean id="productDetailFormController" name="/detail.htm /addToCart.htm" 
        class="detailProduct.DetailProductFormController">
    <property name="sessionForm" value="true" />
    <property name="commandName" value="dpBackObj" />
    <property name="commandClass" value="detailProduct.DetailProductBackingObject" />
    <property name="validator">
        <bean class="detailProduct.DetailProductValidator" />
    </property>
    <property name="formView" value="detail" />
    <property name="successView" value="redirect:/viewCart.htm" />
    <property name="cartService" ref="cartServiceImpl"/>
</bean>

DetailProductFormController的支持对象是:

public class DetailProductBackingObject {
    private String quantityOverflowError;
    private Product product;
    private int quantity;
    private ShoppingCart shoppingCart;
    private long sizeId;
    private long colorId;
    public DetailProductBackingObject() {
        this.product = new Product();
        this.sizeId = -1;
        this.colorId = -1;
    }
    //getters and setters
}

如果您需要其他信息,我会提供。我正在使用Spring 2.5.5。

敬意,
专心

EDIT1由于axtavt的请求 ):

<form:form method="post" commandName="dpBackObj">
<table width="730" border="0" cellspacing="0" cellpadding="0">
    <c:if test="${!empty dpBackObj.quantityOverflowError}">
        <tr>
            <td>
                <c:out value="${dpBackObj.quantityOverflowError}"/>
            </td>
        </tr>
    </c:if>
    <spring:bind path="dpBackObj.*">
        <c:if test="${not empty status.errorMessages}">
            <div class="val-summary text-error" id="errorDivId">
                <div style="" class="val-summary text-error" id="errorDivId">
                    <fmt:message key="detail.error.header"/>
                    <ul>
                        <c:forEach items="${status.errorMessages}" var="error">
                            <li><c:out value="${error}"/></li>
                        </c:forEach>
                    </ul>
                </div>
            </div>
        </c:if>
    </spring:bind>
    <tr>
        <td width="310" align="left" valign="top">
            <img src="${imagesPath}/${dpBackObj.product.largeImageUrl}" alt="${dpBackObj.product.description}" />
        </td>
        <td width="420" align="left" valign="top">
            <div id="tls_detPName">
                <c:out value="${dpBackObj.product.name}"></c:out>
            </div>
            <div >  
                <strong class="numeric">${dpBackObj.product.priceInDollars}</strong>
            </div>
            <div id="tls_detPDescLong">
                ${dpBackObj.product.largeDescription}
                <br />
            </div>
            <div >
                <table cellpadding="2" border="0">
                    <tr>
                        <td align="right">
                            <label for="p_sizes" class="label"><fmt:message key="viewCart.Size"/></label>
                        </td>
                        <td>
                            <form:select path="sizeId" > 
                                <form:option  value="-1" label="x"/> 
                                <form:options items="${dpBackObj.product.sizeMap}"/>
                            </form:select>
                        </td>
                    </tr>
                    <tr>
                        <td align="right">
                            <label for="p_colors" class="label"><fmt:message key="viewCart.Color"/></label>
                        </td>
                        <td>
                            <form:select path="colorId" > 
                                <form:option value="-1" label="y"/> 
                                <form:options items="${dpBackObj.product.colorMap}"/>
                            </form:select>
                        </td>
                    </tr>
                </table>
            </div>
            <div id="tls_addToCart">
                <div >
                    <label for="quantityId" class="label"><fmt:message key="viewCart.Quantity"/>:</label>
                    <form:input path="quantity" onkeypress="return checkForNumber(this, event)" maxlength="10" size="3" id="quantityId" cssClass="textbox-center"/>
                    <input type="image" name="addToCartButtonName" src="${imagesPath}/addToCartBtn.jpg" /> 
                </div>
            </div>
        </td>
    </tr>
</table>
</form:form>

EDIT2由于JacobM的请求 ):这是我的验证器:

public class DetailProductValidator implements Validator {
    public boolean supports(Class clazz) {
        return DetailProductBackingObject.class.equals(clazz);
    }

    public void validate(Object obj, Errors errors) {
        DetailProductBackingObject detailProductBackingObject = (DetailProductBackingObject) obj;
        if (detailProductBackingObject.getSizeId() == -1) {
            errors.rejectValue("sizeId", "error.detail.jsp.choose.size", null, "Input size.");
        }
    }
}

当我到达那一 行时,DetailProductBackingObject detailProductBackingObject = 我已经 遇到
了错误。
将请求参数转换为支持对象属性的过程在http://static.springsource.org/spring/docs/2.5.x/api/org/springframework/web/servlet/mvc/BaseCommandController.html中进行。这是Spring关于转换的说法:

使用请求参数和PropertyEditor进行填充:收到请求后,任何BaseCommandController都将尝试使用请求参数填充命令对象。这是使用典型的众所周知的JavaBeans属性表示法完成的。当存在名为“
firstName”的请求参数时,框架将尝试调用setFirstName([value])并传递参数值。当然支持嵌套属性。例如,名为“
address.city”的参数将导致在命令类上进行getAddress()。setCity([value])调用。

重要的是要意识到,您不仅限于JavaBeans中的String参数。使用java.beans包提供的PropertyEditor-
ionion,您将能够将Strings转换为Objects,反之亦然。例如,对于名为locale的请求参数,其值为en,setLocale(Locale
loc)是完全可能的,只要您在Controller中注册了适当的PropertyEditor(有关此问题的更多信息,请参阅initBinder())。

验证器:控制器成功使用请求中的参数填充命令对象后,它将使用任何已配置的验证器来验证对象。验证结果将放在Errors对象中,该对象可在View中用于呈现任何输入问题。


问题答案:

由于我看不到表单有任何问题,因此我可以想象的唯一可能原因是您product在表单页面的URL中有一个命名的参数。

如果是这样,您可以更改URL或使用DataBinder.setDisallowedFields()禁用绑定该参数的尝试。