JSR-303 @Valid анотація не працює для списку дочірніх об'єктів


95

Мої основні класи - це

public class UserAddressesForm {

    @NotEmpty
    private String firstName;

    @NotEmpty
    private String lastName;

    private List<AddressForm> addresses;

...
setters and getters 

public class AddressForm {

    @NotEmpty
    private String customName;
    @NotEmpty
    private String city;
    @NotEmpty
    private String streetAn;
    @NotEmpty
    private String streetHn;
    @NotEmpty
    private String addressCountry;
    @NotEmpty
    private String postCode;
...
setters and getters

Контролер

@RequestMapping(value = "/up", method = RequestMethod.POST)
    public String completeForm(@Valid @ModelAttribute("userAddressesForm") UserAddressesForm userAddressesForm,  
            BindingResult result, HttpServletRequest req)
...

Сторінка JSP

<form:form commandName="userAddressesForm" action="registered">
    <table>

        <tr>
            <td class="formLabels"><form:label path="firstName">
                <spring:message code="label.name" />
            </form:label></td>
            <td><form:input path="firstName" /></td>
            <td><form:errors path="firstName" cssClass="error" /></td>
        </tr>
        <tr>
            <td class="formLabels"><form:label path="lastName">
                <spring:message code="label.surname" />
            </form:label></td>
            <td><form:input path="lastName" /></td>
            <td><form:errors path="lastName" cssClass="error" /></td>
        </tr>
    </table>

    <c:forEach items="${userAddressesForm.addresses}" varStatus="gridRow">  
        <div id="main_address" class="address_data_form">
            <fieldset>
                <legend><spring:message code="label.stepThreeMainAddressInfo" /></legend>
                <a href="#" class="deleteItem"></a>
                <table>
                    <tr>            
                        <td class="formLabels">
                            <spring:message code="label.address.custom.name" />
                        </td>
                        <td>
                            <spring:bind path="addresses[${gridRow.index}].customName">
                                <input type="input" name="<c:out value="${status.expression}"/>"
                                    id="<c:out value="${status.expression}"/>"
                                    value="<c:out value="${status.value}"/>" />
                                    <form:errors path="${status.expression}"/>
                            </spring:bind>
                        </td>   
                    </tr>               
                    <tr>            
                        <td class="formLabels">
                            <spring:message code="label.streetAnStreetHn" />
                        </td>
                        <td>
                            <spring:bind path="addresses[${gridRow.index}].streetAn">
                                <input type="input" name="<c:out value="${status.expression}"/>"
                                    id="<c:out value="${status.expression}"/>"
                                    value="<c:out value="${status.value}"/>" />
                            </spring:bind>
                            <spring:bind path="addresses[${gridRow.index}].streetHn">
                            <input type="input" name="<c:out value="${status.expression}"/>"
                                id="<c:out value="${status.expression}"/>"
                                value="<c:out value="${status.value}"/>" >
                            <form:errors path="addresses[${gridRow.index}].streetHn"/>
                            </spring:bind>

                        </td>
                    </tr>
                    <tr>                        
                        <td class="formLabels">
                            <spring:message code="label.postCode" />
                        </td>
                        <td>
                            <spring:bind path="addresses[${gridRow.index}].postCode">
                                <input type="input" name="<c:out value="${status.expression}"/>"
                                    id="<c:out value="${status.expression}"/>"
                                    value="<c:out value="${status.value}"/>" />
                            </spring:bind>
                        </td>                   
                    </tr>
                    <tr>                
                        <td class="formLabels">
                            <spring:message code="label.city" />
                        </td>
                        <td>
                            <spring:bind path="addresses[${gridRow.index}].city">
                                <input type="input" name="<c:out value="${status.expression}"/>"
                                    id="<c:out value="${status.expression}"/>"
                                    value="<c:out value="${status.value}"/>" />
                                <form:errors path="addresses[${gridRow.index}].city" cssClass="error" />
                            </spring:bind>
                        </td>
                    </tr>       
                </table>    
            </fieldset>
        </div>
    </c:forEach>

Чому поля об'єктів не перевіряються AddressForm?

Будь ласка, допоможіть.

Відповіді:


165

Вам потрібно прикрасити addressesчлен UserAddressesFormз @Validанотаціями. Див. Розділи 3.1.3 та 3.5.1 JSR 303: Перевірка бобів . Як я пояснив у своїй відповіді на запитання Чи існує стандартний спосіб увімкнути перевірку Bean JSR 303 за допомогою анотованого методу , це справжнє використання @Validанотації відповідно до JSR 303.

Редагувати приклад коду: Hibernate Validator - графік об’єкта . (Список пасажирів в автомобілі)

Редагувати з Hibernate Validator 6 Довідковий документ:

У версіях до 6 Hibernate Validator підтримував каскадну перевірку для підмножини елементів контейнера, і вона була реалізована на рівні контейнера (наприклад, ви могли б використовувати, @Valid private List<Person>щоб увімкнути каскадну перевірку для Person).

Це все ще підтримується, але не рекомендується. Будь ласка, використовуйте @Validзамість них примітки на рівні контейнера, оскільки вони є більш виразними.

Приклад:

public class Car {

        private List<@NotNull @Valid Person> passengers = new ArrayList<Person>();

        private Map<@Valid Part, List<@Valid Manufacturer>> partManufacturers = new HashMap<>();

       //...
   }

Також подивіться, що нового в Bean Validation 2.0 / Jakarta Bean Validation .


А як щодо групування? Я хочу, щоб адреса перевірялася лише тоді, коли група А перевіряється. @Validігнорує групу.
Акшай

1
Анотація @Akshay @Validпрацює на нижчому рівні. Я думаю, що @GroupSequence можна використовувати для вказівки порядку перевірки.
Ritesh

Моє детальне запитання stackoverflow.com/q/41457404/1534925 . Не думаю, що послідовність буде працювати в моєму випадку.
Акшай

27

Додаючи до відповіді @Ritesh, @Validобмеження дасть вказівку Bean Validator заглибитися у тип застосованої властивості та перевірити всі обмеження, знайдені там. Відповідаючи кодом на ваше запитання, валідатор, побачивши @Validобмеження на addressesвластивість, вивчить AddressFormклас і перевірить всі JSR 303обмеження, знайдені всередині, наступним чином:

public class UserAddressesForm {

    @NotEmpty
    private String firstName;

    @NotEmpty
    private String lastName;

    @Valid
    private List<AddressForm> addresses;

...
setters and getters 

public class AddressForm {

    @NotEmpty
    private String customName;
    @NotEmpty
    private String city;
    @NotEmpty
    private String streetAn;
    @NotEmpty
    private String streetHn;
    @NotEmpty
    private String addressCountry;
    @NotEmpty
    private String postCode;
...
setters and getters

1
А як щодо групування? Я хочу, щоб адреса перевірялася лише тоді, коли група А перевіряється. @Valid ігнорує групу
Акшая

Я навіть не прочитав Вашої відповіді, а лише розглянув setters and gettersвирішене моє питання.
Шубхам Кхандаре


4

Робоче рішення.

public class UserAddressesForm {

    @NotEmpty(message="firstName is required")
    private String firstName;

    @NotEmpty(message="lastNameis required")
    private String lastName;

    @NotNull(message="addresses attributes are required")
    @Valid
    private List<AddressForm> addresses;

...
setters and getters 

public class AddressForm {

    @NotEmpty(message="customNameis required")
    private String customName;
    @NotEmpty
    private String city;
    @NotEmpty
    private String streetAn;
    @NotEmpty
    private String streetHn;
    @NotEmpty
    private String addressCountry;
    @NotEmpty
    private String postCode;
...
setters and getters
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.