【SpringMVC】请求参数的绑定

news/2024/11/25 10:48:27/

1.绑定说明

1.1 绑定的机制

我们都知道,表单中请求参数都是基于 key=value 的。SpringMVC 绑定请求参数的过程是通过把表单提交请求参数,作为控制器中方法参数进行绑定的。

例如:
<a href="account/findAccount?accountId=10">查询账户</a>中请求参数是:accountId=10

/**
* 查询账户
* @return
*/
@RequestMapping("/findAccount")
public String findAccount(Integer accountId) {System.out.println("查询了账户。。。。"+accountId);return "success";
}

1.2 支持的数据类型

基本类型参数 :包括基本类型和 String 类型

POJO 类型参数 :包括实体类,以及关联的实体类

数组和集合类型参数 :包括 List 结构和 Map 结构的集合(包括数组)

SpringMVC 绑定请求参数是自动实现的,但是要想使用,必须遵循使用要求。

1.3 使用要求

如果是基本类型或者 String 类型:要求我们的参数名称必须和控制器中方法的形参名称保持一致。(严格区分大小写)

如果是 POJO 类型 ,或者它的关联对象 :要求表单中参数名称和 POJO 类的属性名称保持一致。并且控制器方法的参数类型是 POJO 类型。

如果是集合类型, 有两种方式

  • 第一种:
    要求集合类型的请求参数必须在 POJO 中。在表单中请求参数名称要和 POJO 中集合属性名称相同。
    给 List 集合中的元素赋值,使用下标。
    给 Map 集合中的元素赋值,使用键值对。
  • 第二种:
    接收的请求参数是 json 格式数据。需要借助一个注解实现。

注意:它还可以实现一些数据类型自动转换。内置转换器全都在:org.springframework.core.convert.support包下。有:

java.lang.Boolean -> java.lang.String : ObjectToStringConverter
java.lang.Character -> java.lang.Number : CharacterToNumberFactory
java.lang.Character -> java.lang.String : ObjectToStringConverter
java.lang.Enum -> java.lang.String : EnumToStringConverter
java.lang.Number -> java.lang.Character : NumberToCharacterConverter
java.lang.Number -> java.lang.Number : NumberToNumberConverterFactory
java.lang.Number -> java.lang.String : ObjectToStringConverter
java.lang.String -> java.lang.Boolean : StringToBooleanConverter
java.lang.String -> java.lang.Character : StringToCharacterConverter
java.lang.String -> java.lang.Enum : StringToEnumConverterFactory
java.lang.String -> java.lang.Number : StringToNumberConverterFactory
java.lang.String -> java.util.Locale : StringToLocaleConverter
java.lang.String -> java.util.Properties : StringToPropertiesConverter
java.lang.String -> java.util.UUID : StringToUUIDConverter
java.util.Locale -> java.lang.String : ObjectToStringConverter
java.util.Properties -> java.lang.String : PropertiesToStringConverter
java.util.UUID -> java.lang.String : ObjectToStringConverter
......

如遇特殊类型转换要求,需要我们自己编写自定义类型转换器。

1.4 使用示例

1.基本类型和 String 类型作为参数

jsp代码

<!-- 基本类型示例 -->
<a href="account/findAccount?accountId=10&accountName=zhangsan">查询账户</a>

控制器代码

/**
* 查询账户
* @return
*/
@RequestMapping("/findAccount")
public String findAccount(Integer accountId,String accountName) {System.out.println("查询了账户。。。。"+accountId+","+accountName);return "success";
}

2.POJO 类型作为参数

实体类

public class Account implements Serializable {private Integer id;private String name;private Float money;private Address address;//getters and setters
}
public class Address implements Serializable {private String provinceName;private String cityName;//getters and setters
}

jsp代码

<!-- pojo 类型演示 -->
<form action="account/saveAccount" method="post">账户名称:<input type="text" name="name" ><br/>账户金额:<input type="text" name="money" ><br/>账户省份:<input type="text" name="address.provinceName" ><br/>账户城市:<input type="text" name="address.cityName" ><br/><input type="submit" value=" 保存 ">
</form>

控制器代码

/**
* 保存账户
* @param account
* @return
*/
@RequestMapping("/saveAccount")
public String saveAccount(Account account) {System.out.println("保存了账户。。。。"+account);return "success";
}

3.POJO 类中包含集合类型参数

实体类

public class User implements Serializable {private String username;private String password;private Integer age;private List<Account> accounts;private Map<String,Account> accountMap;//getters and setters@Overridepublic String toString() {return "User [username=" + username + ", password=" + password + ", age="+ age + ",\n accounts=" + accounts + ",\n accountMap=" + accountMap + "]";}
}

jsp代码

<!-- POJO 类包含集合类型演示 -->
<form action="account/updateAccount" method="post">用户名称:<input type="text" name="username" ><br/>用户密码:<input type="password" name="password" ><br/>用户年龄:<input type="text" name="age" ><br/>账户 1 名称:<input type="text" name="accounts[0].name" ><br/>账户 1 金额:<input type="text" name="accounts[0].money" ><br/>账户 2 名称:<input type="text" name="accounts[1].name" ><br/>账户 2 金额:<input type="text" name="accounts[1].money" ><br/>账户 3 名称:<input type="text" name="accountMap['one'].name" ><br/>账户 3 金额:<input type="text" name="accountMap['one'].money" ><br/>账户 4 名称:<input type="text" name="accountMap['two'].name" ><br/>账户 4 金额:<input type="text" name="accountMap['two'].money" ><br/><input type="submit" value=" 保存 ">
</form>

控制器代码

/**
* 更新账户
* @return
*/
@RequestMapping("/updateAccount")
public String updateAccount(User user) {System.out.println("更新了账户。。。。"+user);return "success";
}

4.请求参数乱码问题

post 请求方式

在 web.xml 中配置一个过滤器

<filter><filter-name>CharacterEncodingFilter</filter-name><filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class><init-param><param-name>encoding</param-name><param-value>UTF-8</param-value></init-param><!-- 启动过滤器 --><init-param><param-name>forceEncoding</param-name><param-value>true</param-value></init-param>
</filter>
<!-- 过滤所有请求 -->
<filter-mapping><filter-name>CharacterEncodingFilter</filter-name><url-pattern>/*</url-pattern>
</filter-mapping>

在 SpringMVC的配置文件中可以配置,静态资源不过滤:

<!-- location 表示路径,mapping 表示文件,**表示该目录下的文件以及子目录的文件 -->
<mvc:resources location="/css/" mapping="/css/**"/>
<mvc:resources location="/images/" mapping="/images/**"/>
<mvc:resources location="/scripts/" mapping="/javascript/**"/>

get 请求方式

tomacat 对 GET和 POST 请求处理方式是不同的,GET请求的编码问题,要改 tomcat 的 server.xml配置文件,如下:
<Connector connectionTimeout="20000" port="8080" protocol="HTTP/1.1" redirectPort="8443"/>
改为:
<Connector connectionTimeout="20000" port="8080" protocol="HTTP/1.1" redirectPort="8443" useBodyEncodingForURI="true"/>

如果遇到 ajax 请求仍然乱码,请把:useBodyEncodingForURI="true"改为 URIEncoding="UTF-8"即可。

2.特殊情况

2.1 自定义类型转换器

1.使用场景

jsp代码

<!-- 特殊情况之:类型转换问题 -->
<a href="account/deleteAccount?date=2018-01-01">根据日期删除账户</a>

控制器代码

/**
* 删除账户
* @return
*/
@RequestMapping("/deleteAccount")
public String deleteAccount(String date) {System.out.println("删除了账户。。。。"+date);return "success";
}

运行结果

当我们把控制器中方法参数的类型改为 Date 时:

/**
* 删除账户
* @return
*/
@RequestMapping("/deleteAccount")
public String deleteAccount(Date date) {System.out.println("删除了账户。。。。"+date);return "success";
}

运行结果

异常提示

Failed to convert value of type 'java.lang.String' to required type 'java.util.Date'; nested exception is java.lang.IllegalStateException: Cannot convert value of type 'java.lang.String' to required type 'java.util.Date': no matching editors or conversion strategy found

2.使用步骤

  1. 定义一个类,实现 Converter 接口,该接口有两个泛型。

    public interface Converter<S, T> {//S:表示接受的类型,T:表示目标类型/*** 实现类型转换的方法*/@NullableT convert(S source);
    }
    /**
    * 自定义类型转换器
    */
    public class StringToDateConverter implements Converter<String, Date> {/*** 用于把 String 类型转成日期类型*/@Overridepublic Date convert(String source) {DateFormat format = null;try {if(StringUtils.isEmpty(source)) {throw new NullPointerException("请输入要转换的日期");}format = new SimpleDateFormat("yyyy-MM-dd");Date date = format.parse(source);return date;} catch (Exception e) {throw new RuntimeException("输入日期有误");}}
    }
    
    1. 在 spring 配置文件中配置类型转换器

      spring 配置类型转换器的机制是,将自定义的转换器注册到类型转换服务中去。

      <!-- 配置类型转换器工厂 -->
      <bean id="converterService" class="org.springframework.context.support.ConversionServiceFactoryBean"><!-- 给工厂注入一个新的类型转换器 --><property name="converters"><array><!-- 配置自定义类型转换器 --><bean class="cn.krislin.web.converter.StringToDateConverter"></bean></array></property>
      </bean>
      
    2. 在 annotation-driven 标签中引用配置的类型转换服务

      <!-- 引用自定义类型转换器 --> <mvc:annotation-driven conversion-service="converterService"></mvc:annotation-driven>

2.2 使用 ServletAPI 对象作为方法参数

SpringMVC 还支持使用原始 ServletAPI 对象作为控制器方法的参数。支持原始 ServletAPI 对象有:

HttpServletRequest
HttpServletResponse
HttpSession
java.security.Principal
Locale
InputStream
OutputStream
Reader
Writer

我们可以把上述对象,直接写在控制的方法参数中使用。

jsp代码

<!-- 原始 ServletAPI 作为控制器参数 -->
<a href="account/testServletAPI">测试访问 ServletAPI</a>

控制器代码

/**
* 测试访问 testServletAPI
* @return
*/
@RequestMapping("/testServletAPI")
public String testServletAPI(HttpServletRequest request,HttpServletResponse response,HttpSession session) {System.out.println(request);System.out.println(response);System.out.println(session);return "success";
}

http://www.ppmy.cn/news/6106.html

相关文章

基于防火墙双击热备三层网络规划_ensp综合实验

作者&#xff1a;BSXY_19计科_陈永跃BSXY_信息学院注&#xff1a;未经允许禁止转发任何内容基于防火墙双击热备三层网络规划_ensp综合实验前言及技术/资源下载说明&#xff08; **未经允许禁止转发任何内容** &#xff09;插曲&#xff1a;基于eNSP中大型校园/企业网络规划与设…

机器学习 | 决策树

一.基本原理 决策树是一种树状结构模型&#xff0c;每一个根节点都是一个特征判断&#xff0c;它的叶子节点就是它的特征分类结果 决策树是一种分类和回归的基本模型&#xff0c;是一棵树的形式&#xff0c;其实就是将平时所说的 if-else 语句构建成了树的形式。决策树主要包…

阿里灵杰:与开发者一起推动AI创新落地

对于人工智能领域而言&#xff0c;“AIGC”无疑是贯穿2022年的热点。12月16日&#xff0c;Science杂志发布了2022年度科学十大突破&#xff0c;AIGC赫然在列。以文生图&#xff0c;对话机器人等AI创新应用的落地&#xff0c;引发一轮又一轮的全民狂欢热潮。AI技术蓬勃发展&…

微信小程序之实时聊天系统——页面介绍

目录 系统结果展示&#xff1a; 系统的页面说明&#xff1a; 1.我们首先再app.json中创建四个tabBar页面&#xff08;消息、联系人、用户列表、我的&#xff09; 2.消息页面&#xff1a; 3.联系人页面&#xff1a; 4.用户列表页面&#xff1a; 5.我的页面&#xff1a; 欢…

【C++】PCL对大容量点云进行体素降采样

文章目录0.引言1.普通体素降采样2.OcTree体素降采样0.引言 \qquadPCL的体素滤波器如果对超大容量点云进行降采样&#xff0c;要求降采样后的格点数目必须在整型的范围&#xff0c;不然就会报[pcl::VoxelGrid::applyFilter] Leaf size is too small for the input dataset. Inte…

整理各种Vue项目在IE浏览器白屏报错 SCRIPT1002:语法错误

目录 一、关于 sockjs-client 依赖包 二、关于 highlight 依赖包 三、关于 swiper 依赖包 四、IE 不支持 ES6 语法 五、第三方插件引入导致 六、本地环境正常&#xff0c;生产环境仍旧白屏 这篇文章主要介绍了 Vue 项目在 IE 浏览器显示白屏并报错 SCRIPT1002: 语法错误 …

7、操作系统之间的文件传输

Windows 与 Linux 在Windows与Linux中传输文件的常用方式有两种&#xff0c;一种是使用使用XFTP工具&#xff0c;另一种是使用rz sz 命令的方式进行 lrzsz 安装 yum install lrzsz -y 1&#xff09;rz 将文件从window上传到Linux 2&#xff09;sz 将文件从linux传输到window …

stm32cubemx hal学习记录:电机控制

一、基础配置 1、配置RCC、SYS&#xff0c;SYS的Timebase Source选择TIM6 2、配置USART1、时钟84MHz 3、激活FreeRTOS&#xff0c;选择CMSIS_V1&#xff0c;Config parameters种USE_TIMERS选择ENABLE 二、编码器及电机驱动引脚配置 1、选用TIM3的编码器模式 2、驱动使用L2…