【Spring】基础入门篇(十五) SpringMVC请求参数的绑定

Spring·基础入门篇(十五) SpringMVC请求参数的绑定


1. 绑定的机制

表单中请求参数都是基于 key=value 的。

SpringMVC 绑定请求参数的过程是通过把表单提交请求参数,作为控制器中方法参数进行绑定的。

例如:
在这里插入图片描述

2. 支持的数据类型

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

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

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

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

3. 使用要求

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

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

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

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

SpringMVC内置转换器在 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

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

4. 使用示例

4.1 基本类型和 String 类型作为参数

jsp :

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

控制器:

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

运行结果:
在这里插入图片描述

4.2 POJO 类型作为参数

pojo:

/**
 * @author: LzCc
 * @blog: https://blog.csdn.net/qq_41744145
 * @description: 地址的实体类
 */
public class Address {
    private String provinceName;
    private String cityName;
    // getter、setter...
    // toString...
}
/**
 * @author: LzCc
 * @blog: https://blog.csdn.net/qq_41744145
 * @description: 账户信息
 */
public class Account {
    private Integer id;
    private String name;
    private Float money;
    // getter、setter...
    // toString...
}

jsp:

<!-- pojo 类型演示 -->
<form action="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";
}

运行结果:
在这里插入图片描述

4.3 POJO 类中包含集合类型参数

pojo:

/**
 * @author: LzCc
 * @blog: https://blog.csdn.net/qq_41744145
 * @description:  用户实体类
 */
public class User {
    private String username;
    private String password;
    private Integer age;
    private List<Account> accounts;
    private Map<String,Account> accountMap;
    // getter、setter...
    // toString...
}

jsp:

<!-- POJO 类包含集合类型演示 -->
<form action="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";
}

运行结果:
在这里插入图片描述

5. 请求参数乱码问题

5.1 post 请求方式

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

<!-- 配置 springMVC 编码过滤器 -->
<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/**"/>

5.2 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” 即可。

6. 自定义类型转换器

6.1 使用场景:日期转换

jsp:

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

控制器:

/**
* 删除账户
* @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
意思就是找不到从 String 类型转换到 Date类型的转换器

6.2 解决问题:自定义日期转换器

第一步:定义一个类,实现 Converter 接口,该接口有两个泛型

public interface Converter<S, T> {//S:表示接受的类型,T:表示目标类型
	/**
	* 实现类型转换的方法
	*/
	@Nullable
	T convert(S source);
}
/**
 * @author: LzCc
 * @blog: https://blog.csdn.net/qq_41744145
 * @description: 自定义类型转换器
 */
public class StringToDateConverter implements Converter<String, Date> {
    public 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("输入日期有误");
        }
    }
}

第二步:在 spring 配置文件中配置类型转换器。

<!-- 配置类型转换器工厂 -->
<bean id="converterService" class="org.springframework.context.support.ConversionServiceFactoryBean">
    <!-- 给工厂注入一个新的类型转换器 -->
    <property name="converters">
        <array>
            <!-- 配置自定义类型转换器 -->
            <bean class="top.lzchao.converter.StringToDateConverter"></bean>
        </array>
    </property>
</bean>

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

第三步:在 annotation-driven 标签中引用配置的类型转换服务

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

运行结果:
在这里插入图片描述

7. 使用 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";
}

执行结果:
在这里插入图片描述

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值