`

浅谈 SpringMVC 数据绑定

 
阅读更多
转自: http://senton.iteye.com/blog/973918

查看spring源码可以看出spring支持转换的数据类型:
org.springframework.beans.PropertyEditorRegistrySupport:
	/**
	 * Actually register the default editors for this registry instance.
	 */
	private void createDefaultEditors() {
		this.defaultEditors = new HashMap<Class, PropertyEditor>(64);

		// Simple editors, without parameterization capabilities.
		// The JDK does not contain a default editor for any of these target types.
		this.defaultEditors.put(Charset.class, new CharsetEditor());
		this.defaultEditors.put(Class.class, new ClassEditor());
		this.defaultEditors.put(Class[].class, new ClassArrayEditor());
		this.defaultEditors.put(Currency.class, new CurrencyEditor());
		this.defaultEditors.put(File.class, new FileEditor());
		this.defaultEditors.put(InputStream.class, new InputStreamEditor());
		this.defaultEditors.put(InputSource.class, new InputSourceEditor());
		this.defaultEditors.put(Locale.class, new LocaleEditor());
		this.defaultEditors.put(Pattern.class, new PatternEditor());
		this.defaultEditors.put(Properties.class, new PropertiesEditor());
		this.defaultEditors.put(Resource[].class, new ResourceArrayPropertyEditor());
		this.defaultEditors.put(TimeZone.class, new TimeZoneEditor());
		this.defaultEditors.put(URI.class, new URIEditor());
		this.defaultEditors.put(URL.class, new URLEditor());
		this.defaultEditors.put(UUID.class, new UUIDEditor());

		// Default instances of collection editors.
		// Can be overridden by registering custom instances of those as custom editors.
		this.defaultEditors.put(Collection.class, new CustomCollectionEditor(Collection.class));
		this.defaultEditors.put(Set.class, new CustomCollectionEditor(Set.class));
		this.defaultEditors.put(SortedSet.class, new CustomCollectionEditor(SortedSet.class));
		this.defaultEditors.put(List.class, new CustomCollectionEditor(List.class));
		this.defaultEditors.put(SortedMap.class, new CustomMapEditor(SortedMap.class));

		// Default editors for primitive arrays.
		this.defaultEditors.put(byte[].class, new ByteArrayPropertyEditor());
		this.defaultEditors.put(char[].class, new CharArrayPropertyEditor());

		// The JDK does not contain a default editor for char!
		this.defaultEditors.put(char.class, new CharacterEditor(false));
		this.defaultEditors.put(Character.class, new CharacterEditor(true));

		// Spring's CustomBooleanEditor accepts more flag values than the JDK's default editor.
		this.defaultEditors.put(boolean.class, new CustomBooleanEditor(false));
		this.defaultEditors.put(Boolean.class, new CustomBooleanEditor(true));

		// The JDK does not contain default editors for number wrapper types!
		// Override JDK primitive number editors with our own CustomNumberEditor.
		this.defaultEditors.put(byte.class, new CustomNumberEditor(Byte.class, false));
		this.defaultEditors.put(Byte.class, new CustomNumberEditor(Byte.class, true));
		this.defaultEditors.put(short.class, new CustomNumberEditor(Short.class, false));
		this.defaultEditors.put(Short.class, new CustomNumberEditor(Short.class, true));
		this.defaultEditors.put(int.class, new CustomNumberEditor(Integer.class, false));
		this.defaultEditors.put(Integer.class, new CustomNumberEditor(Integer.class, true));
		this.defaultEditors.put(long.class, new CustomNumberEditor(Long.class, false));
		this.defaultEditors.put(Long.class, new CustomNumberEditor(Long.class, true));
		this.defaultEditors.put(float.class, new CustomNumberEditor(Float.class, false));
		this.defaultEditors.put(Float.class, new CustomNumberEditor(Float.class, true));
		this.defaultEditors.put(double.class, new CustomNumberEditor(Double.class, false));
		this.defaultEditors.put(Double.class, new CustomNumberEditor(Double.class, true));
		this.defaultEditors.put(BigDecimal.class, new CustomNumberEditor(BigDecimal.class, true));
		this.defaultEditors.put(BigInteger.class, new CustomNumberEditor(BigInteger.class, true));

		// Only register config value editors if explicitly requested.
		if (this.configValueEditorsActive) {
			StringArrayPropertyEditor sae = new StringArrayPropertyEditor();
			this.defaultEditors.put(String[].class, sae);
			this.defaultEditors.put(short[].class, sae);
			this.defaultEditors.put(int[].class, sae);
			this.defaultEditors.put(long[].class, sae);
		}
	}



下面挑选一些常用的数据类型,举例说明它们的绑定方式

1. 基本数据类型(以int为例,其他类似):
Controller代码:
	@RequestMapping("test.do")
	public void test(int num) {
		
	}

JSP表单代码:

    <form action="test.do" method="post">  
       <input name="num" value="10" type="text"/>  
       ......  
    </form>


表单中input的name值和Controller的参数变量名保持一致,就能完成基本数据类型的数据绑定,如果不一致可以使用@RequestParam标注实现。值得一提的是,如果Controller方法参数中定义的是基本数据类型,但是从jsp提交过来的数据为null或者""的话,会出现数据转换的异常。也就是说,必须保证表单传递过来的数据不能为null或"",所以,在开发过程中,对可能为空的数据,最好将参数数据类型定义成包装类型,具体参见下面的第二条。

2. 包装类型(以Integer为例,其他类似):
    Controller代码:
    @RequestMapping("test.do")  
    public void test(Integer num) {  
          
    }  


    JSP表单代码:
    <form action="test.do" method="post">  
       <input name="num" value="10" type="text"/>  
       ......  
    </form>  


和基本数据类型基本一样,不同之处在于,JSP表单传递过来的数据可以为null或"",以上面代码为例,如果jsp中num为""或者表单中无num这个input,那么,Controller方法参数中的num值则为null。

3. 自定义对象类型:
    Model代码:
	
    public class User {  
      
        private String firstName;  
      
        private String lastName;  
      
        public String getFirstName() {  
            return firstName;  
        }  
      
        public void setFirstName(String firstName) {  
            this.firstName = firstName;  
        }  
      
        public String getLastName() {  
            return lastName;  
        }  
      
        public void setLastName(String lastName) {  
            this.lastName = lastName;  
        }  
      
    }  


    Controller代码:

    @RequestMapping("test.do")  
    public void test(User user) {  
          
    }  


    JSP表单代码:

    <form action="test.do" method="post">  
       <input name="firstName" value="张" type="text"/>  
       <input name="lastName" value="三" type="text"/>  
       ......  
    </form>  


非常简单,只需将对象的属性名和input的name值一一对应即可。

4. 自定义复合对象类型:
    Model代码:

    public class ContactInfo {  
      
        private String tel;  
      
        private String address;  
      
        public String getTel() {  
            return tel;  
        }  
      
        public void setTel(String tel) {  
            this.tel = tel;  
        }  
      
        public String getAddress() {  
            return address;  
        }  
      
        public void setAddress(String address) {  
            this.address = address;  
        }  
      
    }  
      
    public class User {  
      
        private String firstName;  
      
        private String lastName;  
      
        private ContactInfo contactInfo;  
      
        public String getFirstName() {  
            return firstName;  
        }  
      
        public void setFirstName(String firstName) {  
            this.firstName = firstName;  
        }  
      
        public String getLastName() {  
            return lastName;  
        }  
      
        public void setLastName(String lastName) {  
            this.lastName = lastName;  
        }  
      
        public ContactInfo getContactInfo() {  
            return contactInfo;  
        }  
      
        public void setContactInfo(ContactInfo contactInfo) {  
            this.contactInfo = contactInfo;  
        }  
      
    }  


    Controller代码:

    @RequestMapping("test.do")  
    public void test(User user) {  
        System.out.println(user.getFirstName());  
        System.out.println(user.getLastName());  
        System.out.println(user.getContactInfo().getTel());  
        System.out.println(user.getContactInfo().getAddress());  
    }  


    JSP表单代码:

    <form action="test.do" method="post">  
       <input name="firstName" value="张" /><br>  
       <input name="lastName" value="三" /><br>  
       <input name="contactInfo.tel" value="13809908909" /><br>  
       <input name="contactInfo.address" value="北京海淀" /><br>  
       <input type="submit" value="Save" />  
    </form>  


User对象中有ContactInfo属性,Controller中的代码和第3点说的一致,但是,在jsp代码中,需要使用“属性名(对象类型的属性).属性名”来命名input的name。

5. List绑定:
    List需要绑定在对象上,而不能直接写在Controller方法的参数中。
    Model代码:

    public class User {  
      
        private String firstName;  
      
        private String lastName;  
      
        public String getFirstName() {  
            return firstName;  
        }  
      
        public void setFirstName(String firstName) {  
            this.firstName = firstName;  
        }  
      
        public String getLastName() {  
            return lastName;  
        }  
      
        public void setLastName(String lastName) {  
            this.lastName = lastName;  
        }  
      
    }  
      
           public class UserListForm {  
      
        private List<User> users;  //注意这里要用list, 不能使用数组, 用数组会报错. 同时也表明了为了绑定, 要多定义一个类出来. 可以把它看成是dto
      
        public List<User> getUsers() {  
            return users;  
        }  
      
        public void setUsers(List<User> users) {  
            this.users = users;  
        }  
      
    }  


    Controller代码:

    @RequestMapping("test.do")  
    public void test(UserListForm userForm) {  
        for (User user : userForm.getUsers()) {  
            System.out.println(user.getFirstName() + " - " + user.getLastName());  
        }  
    }  


    JSP表单代码:

    <form action="test.do" method="post">  
       <table>  
          <thead>  
             <tr>  
                <th>First Name</th>  
                <th>Last Name</th>  
             </tr>  
          </thead>  
          <tfoot>  
             <tr>  
                <td colspan="2"><input type="submit" value="Save" /></td>  
             </tr>  
          </tfoot>  
          <tbody>  
             <tr>  
                <td><input name="users[0].firstName" value="aaa" /></td>  
                <td><input name="users[0].lastName" value="bbb" /></td>  
             </tr>  
             <tr>  
                <td><input name="users[1].firstName" value="ccc" /></td>  
                <td><input name="users[1].lastName" value="ddd" /></td>  
             </tr>  
             <tr>  
                <td><input name="users[2].firstName" value="eee" /></td>  
                <td><input name="users[2].lastName" value="fff" /></td>  
             </tr>  
          </tbody>  
       </table>  
    </form>  


其实,这和第4点User对象中的contantInfo数据的绑定有点类似,但是这里的UserListForm对象里面的属性被定义成List,而不是普通自定义对象。所以,在JSP中需要指定List的下标。值得一提的是,Spring会创建一个以最大下标值为size的List对象,所以,如果JSP表单中有动态添加行、删除行的情况,就需要特别注意,譬如一个表格,用户在使用过程中经过多次删除行、增加行的操作之后,下标值就会与实际大小不一致,这时候,List中的对象,只有在jsp表单中对应有下标的那些才会有值,否则会为null,看个例子:
    JSP表单代码:

    <form action="test.do" method="post">  
       <table>  
          <thead>  
             <tr>  
                <th>First Name</th>  
                <th>Last Name</th>  
             </tr>  
          </thead>  
          <tfoot>  
             <tr>  
                <td colspan="2"><input type="submit" value="Save" /></td>  
             </tr>  
          </tfoot>  
          <tbody>  
             <tr>  
                <td><input name="users[0].firstName" value="aaa" /></td>  
                <td><input name="users[0].lastName" value="bbb" /></td>  
             </tr>  
             <tr>  
                <td><input name="users[1].firstName" value="ccc" /></td>  
                <td><input name="users[1].lastName" value="ddd" /></td>  
             </tr>  
             <tr>  
                <td><input name="users[20].firstName" value="eee" /></td>  
                <td><input name="users[20].lastName" value="fff" /></td>  
             </tr>  
          </tbody>  
       </table>  
    </form>  


这个时候,Controller中的userForm.getUsers()获取到List的size为21,而且这21个User对象都不会为null,但是,第2到第19的User对象中的firstName和lastName都为null。打印结果:

    aaa - bbb  
    ccc - ddd  
    null - null  
    null - null  
    null - null  
    null - null  
    null - null  
    null - null  
    null - null  
    null - null  
    null - null  
    null - null  
    null - null  
    null - null  
    null - null  
    null - null  
    null - null  
    null - null  
    null - null  
    null - null  
    eee - fff  



6. Set绑定:
    Set和List类似,也需要绑定在对象上,而不能直接写在Controller方法的参数中。但是,绑定Set数据时,必须先在Set对象中add相应的数量的模型对象。
    Model代码:

    public class User {  
      
        private String firstName;  
      
        private String lastName;  
      
        public String getFirstName() {  
            return firstName;  
        }  
      
        public void setFirstName(String firstName) {  
            this.firstName = firstName;  
        }  
      
        public String getLastName() {  
            return lastName;  
        }  
      
        public void setLastName(String lastName) {  
            this.lastName = lastName;  
        }  
      
    }  
      
    public class UserSetForm {  
      
        private Set<User> users = new HashSet<User>();  
          
        public UserSetForm(){  
            users.add(new User());  
            users.add(new User());  
            users.add(new User());  
        }  
      
        public Set<User> getUsers() {  
            return users;  
        }  
      
        public void setUsers(Set<User> users) {  
            this.users = users;  
        }  
      
    }  


    Controller代码:

    @RequestMapping("test.do")  
    public void test(UserSetForm userForm) {  
        for (User user : userForm.getUsers()) {  
            System.out.println(user.getFirstName() + " - " + user.getLastName());  
        }  
    }  


    JSP表单代码:

    <form action="test.do" method="post">  
       <table>  
          <thead>  
             <tr>  
                <th>First Name</th>  
                <th>Last Name</th>  
             </tr>  
          </thead>  
          <tfoot>  
             <tr>  
                <td colspan="2"><input type="submit" value="Save" /></td>  
             </tr>  
          </tfoot>  
          <tbody>  
             <tr>  
                <td><input name="users[0].firstName" value="aaa" /></td>  
                <td><input name="users[0].lastName" value="bbb" /></td>  
             </tr>  
             <tr>  
                <td><input name="users[1].firstName" value="ccc" /></td>  
                <td><input name="users[1].lastName" value="ddd" /></td>  
             </tr>  
             <tr>  
                <td><input name="users[2].firstName" value="eee" /></td>  
                <td><input name="users[2].lastName" value="fff" /></td>  
             </tr>  
          </tbody>  
       </table>  
    </form>  


基本和List绑定类似。
需要特别提醒的是,如果最大下标值大于Set的size,则会抛出org.springframework.beans.InvalidPropertyException异常。所以,在使用时有些不便。暂时没找到解决方法,如果有网友知道,请回帖共享你的做法。

5. Map绑定:
    Map最为灵活,它也需要绑定在对象上,而不能直接写在Controller方法的参数中。
    Model代码:

    public class User {  
      
        private String firstName;  
      
        private String lastName;  
      
        public String getFirstName() {  
            return firstName;  
        }  
      
        public void setFirstName(String firstName) {  
            this.firstName = firstName;  
        }  
      
        public String getLastName() {  
            return lastName;  
        }  
      
        public void setLastName(String lastName) {  
            this.lastName = lastName;  
        }  
      
    }  
      
    public class UserMapForm {  
      
        private Map<String, User> users;  
      
        public Map<String, User> getUsers() {  
            return users;  
        }  
      
        public void setUsers(Map<String, User> users) {  
            this.users = users;  
        }  
      
    }  


    Controller代码:

    @RequestMapping("test.do")  
    public void test(UserMapForm userForm) {  
        for (Map.Entry<String, User> entry : userForm.getUsers().entrySet()) {  
            System.out.println(entry.getKey() + ": " + entry.getValue().getFirstName() + " - " +  
                                     entry.getValue().getLastName());  
        }  
    }  


    JSP表单代码:

    <form action="test.do" method="post">  
       <table>  
          <thead>  
             <tr>  
                <th>First Name</th>  
                <th>Last Name</th>  
             </tr>  
          </thead>  
          <tfoot>  
             <tr>  
                <td colspan="2"><input type="submit" value="Save" /></td>  
             </tr>  
          </tfoot>  
          <tbody>  
             <tr>  
                <td><input name="users['x'].firstName" value="aaa" /></td>  
                <td><input name="users['x'].lastName" value="bbb" /></td>  
             </tr>  
             <tr>  
                <td><input name="users['y'].firstName" value="ccc" /></td>  
                <td><input name="users['y'].lastName" value="ddd" /></td>  
             </tr>  
             <tr>  
                <td><input name="users['z'].firstName" value="eee" /></td>  
                <td><input name="users['z'].lastName" value="fff" /></td>  
             </tr>  
          </tbody>  
       </table>  
    </form>  


打印结果:

    x: aaa - bbb  
    y: ccc - ddd  
    z: eee - fff  


当controller有两个对象时
model代码
public class Person {
    private String name;

    private int age;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}

public class Dog {
    private String name;
    private int age;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    @Override
    public String toString() {
        return "Dog{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}


jsp代码
<form action="/hello" method="post">
    <input name="person.name" value="wenzhihong"/>
    <input name="person.age" value="32"/>
    <input name="dog.name" value="dog"/>
    <input name="dog.age" value="1"/>
    <input type="submit" value="but1" name="button">
</form>


controller代码
@Controller
public class HelloWorldController {

    @InitBinder("person")
    protected void initBinderPerson(WebDataBinder binder) {
        binder.setFieldDefaultPrefix("person."); //注意这一行
        System.out.println("initBinderPerson ");
    }

    @InitBinder("dog")
    protected void initBinderDog(WebDataBinder binder) {
        binder.setFieldDefaultPrefix("dog."); //注意这一行
        System.out.println("initBinderDog");
    }

    @InitBinder //这个方法可以不要, 写在这里是为了说明, 每个属性的绑定前都会调用这个方法
    protected void initCommon(WebDataBinder binder) {
        System.out.println("initCommon");
    }

    @RequestMapping("/hello")
    public String hello(@ModelAttribute("dog") Dog dog, //注意这里的ModelAttribute里面的名字要跟上面的InitBinder的一致. 这样才知道是绑定哪个
                        @ModelAttribute("person") Person person,//注意这里的ModelAttribute里面的名字要跟上面的InitBinder的一致
                        @RequestParam(value = "button", defaultValue = "button1") String button,
                        Model model) {

        System.out.println("----------------");
        System.out.println(dog.toString());
        System.out.println(person.toString());

        return "helloworld";
    }


也可以不用上面的方法, 定义一个组合类包含 person类及dog类
public class ComClass{
	private Person person;
	private Dog dog;
	
	//get, set 方法
}

这样的话就是多定义一个类出来

分享到:
评论

相关推荐

    SpringMVC数据绑定及数据类型转换

    在SpringMVC中,数据绑定和数据类型转换是两个关键的概念,它们对于构建高效、健壮的Web应用至关重要。 **数据绑定**是SpringMVC中的一种机制,允许我们将用户通过表单或其他方式提交的请求参数自动绑定到控制器中...

    SpringMVC数据绑定入门.rar

    在这个“SpringMVC数据绑定入门”压缩包中,你将找到一系列资源来帮助你理解并掌握SpringMVC的数据绑定机制,这对于任何Java开发者来说都是一个重要的技能。 1. **SpringMVC概述**: SpringMVC作为Spring框架的一...

    SpringMVC 数据绑定实例详解

    本篇将深入讲解 SpringMVC 数据绑定的实例详解。 SpringMVC 的数据绑定是通过 `ModelAttribute` 注解和 `@RequestParam` 注解来实现的。`@ModelAttribute` 通常用于控制器方法的参数,它可以从模型(Model)中获取...

    浅谈SpringMVC HandlerInterceptor诡异问题排查

    浅谈SpringMVC HandlerInterceptor诡异问题排查 SpringMVC中的HandlerInterceptor是非常重要的组件之一,它可以在请求处理的各个阶段进行干预和修改。本文将主要介绍如何排查SpringMVC HandlerInterceptor中的诡异...

    SpringMVC入门很简单之数据绑定(下)

    在本篇中,我们将深入探讨SpringMVC框架中的数据绑定,这是Java开发中一个至关重要的概念,尤其是在构建Web应用程序时。SpringMVC是Spring框架的一部分,它为开发RESTful服务和传统的表单提交提供了强大的支持。数据...

    Spring MVC注解与数据绑定(含源码及结果图)

    1、有如下一个订单信息页面order.jsp(置于/WEB-INF/jsp目录下),按以下步骤实现一个使用POJO类型完成表单数据传输的SpringMVC数据绑定项目。 (1) 创建一个Order类来封装上述订单信息,其中各个属性的名称和数据类型...

    Spring MVC数据绑定概述及原理详解

    简单数据绑定通常涉及到Java的基本数据类型,例如`int`、`String`、`Double`等。在下面的例子中,我们看到`Integer id`参数被绑定到请求参数`id`: ```java @RequestMapping(value="/selectUser") public String ...

    springmvc入门参数绑定ssm整合

    SpringMVC入门参数绑定SSM整合 SpringMVC是一种基于MVC模式的Web应用程序框架,它是Spring框架的一个模块,用于开发Web项目。下面是SpringMVC入门参数绑定SSM整合的知识点总结: 一、SpringMVC概述 SpringMVC是...

    浅谈Springmvc中的页面跳转问题

    浅谈Springmvc中的页面跳转问题是指在SpringMvc框架中,Controller处理完数据后如何将用户重定向到另一个页面或功能处理方法的过程。这个问题是SpringMvc中一个非常重要的知识点,需要开发者对其有深入的了解。 ...

    SpringMVC绑定数据库例子

    SpringMVC是Spring框架的一部分,它提供了处理HTTP请求、视图解析以及模型数据绑定等功能,而MyBatis则是一个轻量级的持久层框架,它简化了SQL操作,使开发者可以直接编写SQL语句并与Java对象进行映射。 1. **...

    SpringMVC入门很简单之数据绑定(上)

    在本文中,我们将深入探讨SpringMVC框架中的数据绑定,这是Spring MVC开发中不可或缺的一部分,尤其是在处理用户输入和展示响应时。数据绑定是将HTTP请求参数与Java对象的属性关联的过程,极大地简化了Web应用程序的...

    扩展SpringMVC以支持更精准的数据绑定1

    在“扩展SpringMVC以支持更精准的数据绑定1”这个主题中,博主探讨了如何通过自定义转换器和验证器来增强Spring MVC的数据绑定能力,以满足更为复杂的应用场景。 首先,我们要了解Spring MVC的数据绑定基础。在默认...

    SpringMVC参数绑定

    在SpringMVC框架中,参数绑定是核心概念之一,主要负责从前端接收数据,并将其绑定到后端Controller层的方法参数上。这个过程是SpringMVC处理请求和响应的重要环节,涉及到前端和后端的数据交互。 首先,参数绑定...

    Springmvc前后台数据数组绑定

    Springmvc前后台数据数组绑定,list超过了256时,会报错,如何解决?该文给出了说明

    springmvc-demo05-数据绑定(接受参数).zip

    在这个名为"springmvc-demo05-数据绑定(接受参数)"的项目中,我们将深入探讨Spring MVC如何实现这一特性。 1. **数据绑定的基本概念** 数据绑定是将用户输入或者HTTP请求参数自动映射到Java对象的属性上的过程。...

    SpringMVC数据类型转换超详细介绍

    SpringMVC自Spring 3版本开始,对数据绑定机制进行了大幅度改进,提供了更为强大和灵活的数据转换与验证功能。本文将详细介绍SpringMVC中数据类型转换的具体实现方式及其背后的工作原理。 #### 二、Spring 3之前的...

    SpringMVC配置多数据源实战

    在SpringMVC框架中配置多数据源是一项常见的需求,尤其在大型企业级应用中,由于业务的复杂性,往往需要连接不同的数据库以满足不同模块的需求。以下将详细讲解如何实现这一功能。 首先,理解数据源(DataSource)...

    springmvc数据验证

    如果验证失败,Spring会将错误信息自动绑定到模型中,供视图层展示。 2. **自定义Validator**:对于更复杂或者标准注解无法满足的验证需求,可以实现Spring的`Validator`接口,创建自定义的验证器。在这个接口中,...

Global site tag (gtag.js) - Google Analytics