以前,学习过JavaWeb的朋友都知道,我们获取请求参数的代码基本上就是,写一个自己的servlet类,然后继承HttpServlet,再写一个doGet、doPost方法(当然自己随便写一个其他方法也行),然后用上形参HttpServletRequest req,通过req.getParameter方法获取单个唯一的请求参数(比如:账号)或req.getParameterValues方法获取多个同名的请求参数(比如:复选框)。
例如:
但是,现在我们运用SpringMVC的框架了,不需要手写每一个Servlet,自然写法也不同了。那么,SpringMVC怎么获取请求参数呢?请看下文!
我们知道,其实我们每一个RequestMapping注解的方法就是一个servlet,那什么才算servlet呢?在JavaWeb中,我们得实现Servlet接口,或者继承HttpServlet类(这个类其实自己实现了Servlet接口),然后再到配置文件中去注册这个Servlet才算是一个servlet。
现在,我们只需要使用一个RequestMapping注解,就等价于这就是一个servlet。前面也说了,servlet获取请求参数,可以自己随便写一个方法,然后给他附上形参HttpServletRequest req,通过req.getParameter方法获取单个唯一的请求参数(比如:账号)或req.getParameterValues方法获取多个同名的请求参数(比如:复选框)。
那么,我们现在可不可以也直接在我们用了RequestMapping注解的方法中用这个形参来获取参数呢?答案是肯定的。而且这就是这个小节要讲的调用ServletAPI获取请求参数。
代码如下:
我们先写一个页面param.html
DOCTYPE html>
<html lang="en" xmlns:th="http://www.thymeleaf.org">
<head>
<meta charset="UTF-8">
<title>获取请求参数title>
head>
<body>
<h1>测试获取请求参数h1>
body>
html>
然后写我们的控制器方法:
package com.example.controller;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import javax.servlet.http.HttpServletRequest;
@Controller
public class ParamController {
@RequestMapping("/param")
//调用ServletAPI
public String toParam(HttpServletRequest request){
//获取请求参数username
String username = request.getParameter("username");
System.out.println("username = "+username);
return "param";
}
}
然后重启我们的服务器:
并输入地址:http://localhost:8080/mvcDemo/param?username=Keeling
可以看到我们的后台控制器显示如下:
Keeling我经常用,是不是显得好像这个名字是个bug,换其他的行不行,当然行,我再来乱写一个:
看看后台怎么办(有点缺心眼哈),显然,他还是成功获取到了:
上面那种写法,其实还是有点麻烦,又是调用接口又是写什么getParameter的,还要拿个字符串啥的去给他盛。有没有更简单的方法呀。当然有,其实我们只需要在方法里面直接用相同的参数名就可以了。说得有点含糊?没关系,看代码就懂了:
@RequestMapping("/controllerParam")
//写对参数名就行
public String toPram_2(String username, String password){
System.out.println("username = "+username);
System.out.println("password = "+password);
return "param";
}
然后重启服务器:
输入地址:http://localhost:8080/mvcDemo/controllerParam?username=hahaha&password=hello
控制台显示:
那如果我们的password不写呢?(那password会被赋值null)
那如果,我写别的参数呢(答案是,当然不会输出那个别的参数)
到这里有些人可能会疑问了,那怎么让客户准确输入username=什么&password=什么呢?拜托,客户是什么,客户是傻X啊,你怎么可以让他来写,当然是自己写死username=value1,password=value2,然后再帮客户把他填的数据放到value1和value2中去啦。
这个时候,又有问题了,那如果是多选呢,比如hobby可以有多个,那怎么搞?简单啦,还是直接用一个String给他拿捏住。瞧代码:
@RequestMapping("/controllerParam")
public String toPram_2(String username, String password, String hobbies){
System.out.println("username = "+username);
System.out.println("password = "+password);
System.out.println("hobbies = "+hobbies);
return "param";
}
访问地址:http://localhost:8080/mvcDemo/controllerParam?username=hahaha&password=123456&hobbies=c&hobbies=java
此时,控制台显示:(同名的也被成功获取了,然后用逗号,隔开)
可是这样写好像有点傻傻的。。。。行吧,真麻烦,我们再改一下,这次我们用String[]来接这个复数的参数:
@RequestMapping("/controllerParam")
public String toPram_2(String username, String password, String[] hobbies){
System.out.println("username = "+username);
System.out.println("password = "+password);
System.out.println(Arrays.toString(hobbies));
return "param";
}
@RequestParam注解是将请求参数和控制器方法的形参创建映射关系的注解。他一共有三个属性:
1、value:指定为形参赋值的请求参数的参数名
2、required:设置是否必须传输此请求参数,默认值为true
3、defaultValue:不管required属性为true或false,当value所指定的请求参数没有传输或传输的值为“”时,则使用默认值为形参赋值。(比较常用)
直接来写个例子吧:
@RequestMapping("/controllerParam")
public String toPram_2(
@RequestParam(value = "user_name", required = false, defaultValue = "张三") String username,
String password,
String[] hobbies){
System.out.println("username = "+username);
System.out.println("password = "+password);
System.out.println(Arrays.toString(hobbies));
return "param";
}
按以上这么设置后,我们访问的时候username就不能叫username了,而要用注解中value的名称user_name,required = false指明这个值可以不写,defaultValue则设置了若不写user_name或给user_name设置为“”的时候,用张三作为他的值。
访问一下网址:http://localhost:8080/mvcDemo/controllerParam?username=hahaha&password=123456&hobbies=c&hobbies=java
我们可以看到后台输出:
为什么输出张三了呢?答案其实很简单,因为我们用的是username,所以user_name的值就相当于我们没有写。
不信,我们再修改一下访问地址:http://localhost:8080/mvcDemo/controllerParam?user_name=hahaha&password=123456&hobbies=c&hobbies=java
@RequestHeader注解 和 @CookieValue注解 的属性和用法跟@RequestParam相同,不同的是
@RequestHeader注解是将请求头信息和控制器方法的形参创建映射关系
@CookieValue注解是将cookie数据和控制器方法的形参创建映射关系
可以在控制器方法的形参位置设置一个实体类类型的形参,此时若浏览器传输的请求参数的参数名和实体类中的属性名一致,那么请求参数就会为此属性赋值。
比如,我们有一个html代码如下:
pojo.html
DOCTYPE html>
<html lang="en" xmlns:th="http://www.thymeleaf.org">
<head>
<meta charset="UTF-8">
<title>POJO测试title>
head>
<body>
<form th:action="@{/testPOJO}" method="post">
用户名:<input type="text" name="username">
<br>
密 码:<input type="password" name="password">
<br>
性 别:<input type="radio" name="gender" value="boy">男
<input type="radio" name="gender" value="girl">女
<br>
年 龄:<input type="number" name="age">
<br>
邮 箱:<input type="text" name="email">
<br>
<input type="submit" value="提交">
form>
body>
html>
然后就写一个实体类代码如下:
User.java
package com.example.po;
public class User {
private String username;
private String password;
private String gender;
private Integer age;
private String email;
public User() {
}
public User(String username, String password, String gender, Integer age, String email) {
this.username = username;
this.password = password;
this.gender = gender;
this.age = age;
this.email = email;
}
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
public String getPassword() {
return password;
}
public void setPassword(String password) {
this.password = password;
}
public String getGender() {
return gender;
}
public void setGender(String gender) {
this.gender = gender;
}
public Integer getAge() {
return age;
}
public void setAge(Integer age) {
this.age = age;
}
public String getEmail() {
return email;
}
public void setEmail(String email) {
this.email = email;
}
@Override
public String toString() {
return "User{" +
"username='" + username + '\'' +
", password='" + password + '\'' +
", gender='" + gender + '\'' +
", age=" + age +
", email='" + email + '\'' +
'}';
}
}
最后写一个控制器方法(直接通过这个类对象把整个form表格的内容都提取出来):
@RequestMapping("/testPOJO")
public String testPOJO(User user){
System.out.println(user);
return "pojo";
}
输入指定的值之后,点击提交按钮
后台控制器显示如下:
以上就是传说中的使用POJO获取请求参数。
这里,我们如果有中文的话,其实是会出现乱码的,如下:
结果:
那这个中文乱码问题怎么解决呢?请看,
首先在我们的web.xml中配置过滤器,代码如下:
<filter>
<filter-name>CharacterEncodingFilterfilter-name>
<filter-class>org.springframework.web.filter.CharacterEncodingFilterfilter-class>
<init-param>
<param-name>encodingparam-name>
<param-value>UTF-8param-value>
init-param>
<init-param>
<param-name>forceResponseEncodingparam-name>
<param-value>trueparam-value>
init-param>
filter>
<filter-mapping>
<filter-name>CharacterEncodingFilterfilter-name>
<url-pattern>/*url-pattern>
filter-mapping>
我们来看看他的底层(hia hia hia hia):
在IDEA中按住 Ctrl 然后点击下图框住的地方进去可以看到他的源码
上面我的代码改的就是这两个东西
那么真正修改编码格式的代码在哪里呢?我们往下拉,找到doFilterInternal这个方法。在这里,我们可以清晰地看到request和response调用了setCharacterEncoding方法来设置这个编码格式。
为了让他设置我们既要满足他的两个if条件:
先看request的
他的isForceRequestEncoding方法,如果没有设置forceRequestEncoding的话,默认返回的是false。
详细可以看这个无参构造:
因此,我上边的做法是满足if的第二个条件(因为他是或,满足一个即可):
怎么满足呢?很简单,不理他就行了。为啥子嘞?因为不理他,request.getCharacterEncoding()得出来的才会是null。if条件就满足了。
那第二个if语句要怎么满足呢?
也简单啊,设置forceResponseEncoding为true不就行了。
如此,便可以把客户端和服务器的编码格式都变为UTF-8了。
当然还有另一种写法,我们看它里面有一个setter注入的方法如下:
我们只要设置这个forceEncoding为true就可以同时满足上面说到的两个if语句了。
所以,我们web.xml的语句也可以用:
<filter>
<filter-name>CharacterEncodingFilterfilter-name>
<filter-class>org.springframework.web.filter.CharacterEncodingFilterfilter-class>
<init-param>
<param-name>encodingparam-name>
<param-value>UTF-8param-value>
init-param>
<init-param>
<param-name>forceEncodingparam-name>
<param-value>trueparam-value>
init-param>
filter>
<filter-mapping>
<filter-name>CharacterEncodingFilterfilter-name>
<url-pattern>/*url-pattern>
filter-mapping>
补充一下,这个encoding为什么也要设置。这个框架是外国人搞出来的,你不设置,外国佬的框架怎么知道你要用的是UTF-8呢?而且,你不设置的话,他就是null,两个if条件满足后,传参传一个null就去。。。。这礼貌吗???