Подтвердить что ты не робот

Spring Использование MVC формы: флажок для привязки данных

Я знаю, что уже были вопросы по этой теме, но я не понял, как решить следующую проблему:

У меня есть отношение пользователя/ролей, и я хочу перечислить все доступные роли в JSP как элементы флажка, где выбраны флажки, назначенные пользователям. Однако совпадающие элементы не проверяются (здесь я использую Spring 3.1).

Извлечь из объекта User:

private Set<RoleEntity> roles = new HashSet<RoleEntity>();

Извлечь из контроллера Spring (добавление объекта пользователя и списка ролей в модель):

UserEntity userEntity = userEntityService.findById(UserEntity.class, new Long(id));
model.addAttribute("userAttribute", userEntity);

List<RoleEntity> roleList = roleEntityService.findAll();
model.addAttribute("roleList", roleList);

Извлечь из JSP:

<form:form modelAttribute="userAttribute" method="POST" action="${saveUrl}">
...

    <table align="center">
        <tr>
            <td>ID</td>
            <td>Role Name</td>
        </tr>
        <c:forEach items="${roleList}" var="role" varStatus="status">
            <tr>
                <td><form:checkbox path="roles" value="${role}" label="${role.id}" /></td>
                <td><c:out value="${role.name}" /></td>
            </tr>
        </c:forEach>
    </table>

...
</form:form>

Документация MVC Spring гласит: Когда связанное значение имеет тип array или java.util.Collection, вход (checkbox) помечен как "checked", если настроенное значение setValue (Object) присутствует в связанной коллекции.

Разве это не так? Что мне здесь не хватает?

Большое спасибо.

Пол

4b9b3361

Ответ 1

Мое предположение вам не хватает реализации методов equals и hashcode в классе RoleEntity.

Когда связанное значение имеет тип array или java.util.Collection, вход (флажок) помечен как "checked", если настроенное значение setValue (Object) присутствует в связанной коллекции.

Это правильно, но для проверки наличия в HashSet вам нужно выполнить equals и hashcode правильно.

Как быстрый тест, чтобы увидеть, если эта проблема, замените эту строку:

model.addAttribute("roleList", roleList);

с этой строкой:

model.addAttribute("roleList", userEntity.getRoles());

Вы проверяете все свои флажки? Если да, то вы не предоставили свои собственные equals и hashcode, а используются по умолчанию (те, которые унаследованы от Object).

По умолчанию equals сравнивает идентификатор, что означает, что переменная содержит тот же экземпляр, что и другая переменная. Равенство означает, что два разных объекта содержат одно и то же состояние или имеют одинаковый смысл, так сказать.

Использование model.addAttribute("roleList", userEntity.getRoles()) запускает метод по умолчанию equals для возврата true, потому что список и значения, которые вы проверяете на наличие в списке, идентичны (два одинаковых объекта всегда равны).

Но в вашем случае вы используете userEntityService.findById для одного и roleEntityService.findAll для другого, что означает разные объекты. На этом этапе вы должны использовать правильный тест равенства, а не идентичность.

Есть ли у вас equals/hashcode?

На основе вашего кода приведен пример, который работает:

Контроллер:

import java.util.Arrays;
import java.util.Collections;
import java.util.HashSet;
import java.util.List;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;

@Controller
public class SomeController {
    @RequestMapping(value = "/something", method = { RequestMethod.GET, RequestMethod.POST })
    public String handle(Model model) {

        UserEntity userEntity = new UserEntity();
        userEntity.setRoles(new HashSet<RoleEntity>());
        Collections.addAll(userEntity.getRoles(), 
                                new RoleEntity(1, "one"), 
                                new RoleEntity(3, "three"));
        model.addAttribute("userAttribute", userEntity);        

        List<RoleEntity> roleList = Arrays.asList(
                                        new RoleEntity(1, "one"), 
                                        new RoleEntity(2, "two"), 
                                        new RoleEntity(3, "three")
                                    );
        model.addAttribute("roleList", roleList);

        return "view";
    }
}

Класс пользователя:

import java.util.HashSet;
import java.util.Set;

public class UserEntity {
    private Set<RoleEntity> roles = new HashSet<RoleEntity>();

    public Set<RoleEntity> getRoles() {
        return roles;
    }
    public void setRoles(Set<RoleEntity> roles) {
        this.roles = roles;
    }
}

Класс ролей ( обратите внимание на методы equals и hashcode; если вы их удалите, пример больше не работает):

public class RoleEntity {
    private long id;
    private String name;

    @Override
    public int hashCode() {
        return new Long(id).hashCode();
    }

    @Override
    public boolean equals(Object obj) {
        if (obj == null) {
            return false;
        }
        if (! (obj instanceof RoleEntity)) {
            return false;
        }
        return this.id == ((RoleEntity)obj).getId();
    }

    public RoleEntity(long id, String name) {
        this.id = id;
        this.name = name;
    }

    public long getId() {
        return id;
    }
    public void setId(long id) {
        this.id = id;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
}

Вид:

<%@ page language="java" contentType="text/html; charset=ISO-8859-1" pageEncoding="ISO-8859-1"%>
<%@ taglib prefix="form" uri="http://www.springframework.org/tags/form" %>
<%@taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
<form:form modelAttribute="userAttribute" method="POST" action="/something">
    <table align="center">
        <tr>
            <td>ID</td>
            <td>Role Name</td>
        </tr>
        <c:forEach items="${roleList}" var="role">
            <tr>
                <td><form:checkbox path="roles" value="${role}" label="${role.id}" /></td>
                <td><c:out value="${role.name}" /></td>
            </tr>
        </c:forEach>
    </table>
</form:form>

P.S. Только одно замечание о вашем JSP.. Если вы делаете value="${role}" для своей формы: флажок, вы получите атрибуты флажка HTML, такие как value="[email protected]", которые впоследствии могут вызвать вас в другой проблеме.