SpringMVC中请求传输的各种类型参数分析

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
全局流量管理 GTM,标准版 1个月
简介: SpringMVC中请求传输的各种类型参数分析

【1】基本数据类型(以int为例,其他类似)

Controller代码

@RequestMapping("saysth.do")
public void test(int count) {
}

表单代码:

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


表单中input的name值和Controller的参数变量名保持一致,就能完成数据绑定,如果不一致可以使用@RequestParam注解。


需要注意的是,如果Controller方法参数中定义的是基本数据类型,但是从页面提交过来的数据为null或者" "的话,会出现数据转换的异常。


也就是必须保证表单传递过来的数据不能为null或" ",所以,在开发过程中,对可能为空的数据,最好将参数数据类型定义成包装类型,具体参见下面的例子。


当前情况将会按照@RequestParam进行解析。

【2】包装类型(以Integer为例,其他类似)

Controller代码:

@RequestMapping("saysth.do")
public void test(Integer count) {
}

表单代码:

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


和基本数据类型基本一样,不同之处在于,表单传递过来的数据可以为null" ",以上面代码为例,如果表单中null" "或者表单中无num这个input,那么,Controller方法参数中的num值则为null。当前情况将会按照@RequestParam进行解析。


【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("saysth.do")
public void test(User user) {
}


表单代码:

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


非常简单,只需将对象的属性名和input的name值一一匹配即可。后台这里会按照@ModelAttribute注解进行解析。

【4】自定义复合对象类型

Model代码:


User 中包含ContactInfo 。

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("saysth.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());
}

表单代码:

<form action="saysth.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点说的一致,但是,在表单代码中,需要使用“属性名(对象类型的属性).属性名”来命名input的name。后台这里会按照@ModelAttribute注解进行解析。


【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;
    public List<User> getUsers() {
        return users;
    }
    public void setUsers(List<User> users) {
        this.users = users;
    }
}


Controller代码:

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


表单代码:

<form action="saysth.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,而不是普通自定义对象。


所以,在表单中需要指定List的下标。值得一提的是,Spring会创建一个以最大下标值为size的List对象,所以,如果表单中有动态添加行、删除行的情况,就需要特别注意。譬如一个表格,用户在使用过程中经过多次删除行、增加行的操作之后,下标值就会与实际大小不一致。这时候,List中的对象,只有在表单中对应有下标的那些才会有值,否则会为null,看个例子。表单代码:

<form action="saysth.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>();
  //set中添加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("saysth.do")
public void test(UserSetForm userForm) {
    for (User user : userForm.getUsers()) {
        System.out.println(user.getFirstName() + " - " + user.getLastName());
    }
}


表单代码:

<form action="saysth.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异常。所以,在使用时有些不便。


【7】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("saysth.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());
    }
}


表单代码:

<form action="saysth.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


目录
相关文章
|
14天前
|
缓存 前端开发 Java
【Java面试题汇总】Spring,SpringBoot,SpringMVC,Mybatis,JavaWeb篇(2023版)
Soring Boot的起步依赖、启动流程、自动装配、常用的注解、Spring MVC的执行流程、对MVC的理解、RestFull风格、为什么service层要写接口、MyBatis的缓存机制、$和#有什么区别、resultType和resultMap区别、cookie和session的区别是什么?session的工作原理
【Java面试题汇总】Spring,SpringBoot,SpringMVC,Mybatis,JavaWeb篇(2023版)
|
1天前
|
XML 缓存 前端开发
springMVC02,restful风格,请求转发和重定向
文章介绍了RESTful风格的基本概念和特点,并展示了如何使用SpringMVC实现RESTful风格的请求处理。同时,文章还讨论了SpringMVC中的请求转发和重定向的实现方式,并通过具体代码示例进行了说明。
springMVC02,restful风格,请求转发和重定向
|
1月前
|
Java 数据库连接 Spring
后端框架入门超详细 三部曲 Spring 、SpringMVC、Mybatis、SSM框架整合案例 【爆肝整理五万字】
文章是关于Spring、SpringMVC、Mybatis三个后端框架的超详细入门教程,包括基础知识讲解、代码案例及SSM框架整合的实战应用,旨在帮助读者全面理解并掌握这些框架的使用。
后端框架入门超详细 三部曲 Spring 、SpringMVC、Mybatis、SSM框架整合案例 【爆肝整理五万字】
|
1月前
|
XML JSON 数据库
SpringMVC入门到实战------七、RESTful的详细介绍和使用 具体代码案例分析(一)
这篇文章详细介绍了RESTful的概念、实现方式,以及如何在SpringMVC中使用HiddenHttpMethodFilter来处理PUT和DELETE请求,并通过具体代码案例分析了RESTful的使用。
SpringMVC入门到实战------七、RESTful的详细介绍和使用 具体代码案例分析(一)
|
1月前
|
前端开发 Java Spring
SpringMVC种通过追踪源码查看是哪种类型的视图渲染器(一般流程方法)
这篇文章通过示例代码展示了如何在Spring MVC中编写和注册拦截器,以及如何在拦截器的不同阶段添加业务逻辑。
SpringMVC种通过追踪源码查看是哪种类型的视图渲染器(一般流程方法)
|
1月前
|
前端开发 应用服务中间件 数据库
SpringMVC入门到实战------八、RESTful案例。SpringMVC+thymeleaf+BootStrap+RestFul实现员工信息的增删改查
这篇文章通过一个具体的项目案例,详细讲解了如何使用SpringMVC、Thymeleaf、Bootstrap以及RESTful风格接口来实现员工信息的增删改查功能。文章提供了项目结构、配置文件、控制器、数据访问对象、实体类和前端页面的完整源码,并展示了实现效果的截图。项目的目的是锻炼使用RESTful风格的接口开发,虽然数据是假数据并未连接数据库,但提供了一个很好的实践机会。文章最后强调了这一章节主要是为了练习RESTful,其他方面暂不考虑。
SpringMVC入门到实战------八、RESTful案例。SpringMVC+thymeleaf+BootStrap+RestFul实现员工信息的增删改查
|
1月前
|
JSON 前端开发 Java
Spring MVC返回JSON数据
综上所述,Spring MVC提供了灵活、强大的方式来支持返回JSON数据,从直接使用 `@ResponseBody`及 `@RestController`注解,到通过配置消息转换器和异常处理器,开发人员可以根据具体需求选择合适的实现方式。
94 4
|
1月前
|
XML 前端开发 Java
Spring MVC接收param参数(直接接收、注解接收、集合接收、实体接收)
Spring MVC提供了灵活多样的参数接收方式,可以满足各种不同场景下的需求。了解并熟练运用这些基本的参数接收技巧,可以使得Web应用的开发更加方便、高效。同时,也是提高代码的可读性和维护性的关键所在。在实际开发过程中,根据具体需求选择最合适的参数接收方式,能够有效提升开发效率和应用性能。
83 3
|
1月前
|
XML 前端开发 Java
Spring MVC接收param参数(直接接收、注解接收、集合接收、实体接收)
Spring MVC提供了灵活多样的参数接收方式,可以满足各种不同场景下的需求。了解并熟练运用这些基本的参数接收技巧,可以使得Web应用的开发更加方便、高效。同时,也是提高代码的可读性和维护性的关键所在。在实际开发过程中,根据具体需求选择最合适的参数接收方式,能够有效提升开发效率和应用性能。
83 2
|
2月前
|
前端开发 Java 应用服务中间件
我以为我对Spring MVC很了解,直到我遇到了...
所有人都知道Spring MVC是是开发的,却鲜有人知道Spring MVC的理论基础来自于1978 年提出MVC模式的一个老头子,他就是Trygve Mikkjel Heyerdahl Reenskaug,挪威计算机科学家,名誉教授。Trygve Reenskaug的MVC架构思想早期用于图形用户界面(GUI) 的软件设计,他对MVC是这样解释的。MVC 被认为是解决用户控制大型复杂数据集问题的通用解决方案。最困难的部分是为不同的架构组件想出好的名字。模型-视图-编辑器是第一个。
111 1
我以为我对Spring MVC很了解,直到我遇到了...