Spring MVC usage of form:checkbox to bind data

后端 未结 1 617
没有蜡笔的小新
没有蜡笔的小新 2020-12-07 19:15

I know there have been already questions around this topic, but I have not figured out how to solve the following issue:

I have a user/roles relationship and I want

相关标签:
1条回答
  • 2020-12-07 20:15

    My guess is you are missing the implementation for the equals and hashcode methods on the RoleEntity class.

    When the bound value is of type array or java.util.Collection, the input(checkbox) is marked as 'checked' if the configured setValue(Object) value is present in the bound Collection.

    This is correct, but to check for presence in a HashSet you need equals and hashcode implemented correctly.

    Just as a quick test to see if that's the problem, replace this line:

    model.addAttribute("roleList", roleList);
    

    with this line:

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

    Do you get all your checkboxes checked? If yes, then you didn't provide your own equals and hashcode and the default ones are used (those inherited from Object).

    The default equals compares identity which means that a variable holds the same instance as another variable. Equality means two different object contain the same state or have the same meaning, so to speak.

    Using model.addAttribute("roleList", userEntity.getRoles()) triggers the default equals method to return true because the list and values you check for presence in the list are identical (two identical objects are always equal).

    But in your case you use userEntityService.findById for one and roleEntityService.findAll for the other which means different objects. At this point you have to use a proper equality test as opposed to identity.

    Do you have equals/hashcode implemented?

    Based on your code, here is an example that works:

    Controller:

    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";
        }
    }
    

    User class:

    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;
        }
    }
    

    Roles class (notice the equals and hashcode methods; if you remove them, the example no longer works):

    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;
        }
    }
    

    View:

    <%@ 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. Just one observation about your JSP. If you do value="${role}" for your form:checkbox you will get HTML checkbox attributes like value="your.pack.age.declaration.RoleEntity@1" which might get you in another sort of trouble later on.

    0 讨论(0)
提交回复
热议问题