SpringMVC 学习笔记 Part2
1. 请求参数的绑定
绑定机制:表单提交的数据都是key-value格式的,如username=root&password=123。 SpringMVC的参数绑定过程是把表单提交的请求参数,作为控制器中方法的参数进行绑定的,要求提交表单的name和参数的名称是相同的。
支持的数据类型:① 基本数据类型和字符 ② 实体类型(JavaBean)③ 集合数据类型(List、map集合等)。
语法格式:我们表单中参数名称必须和控制器中方法的形参名称保持一致。
基本数据类型和字符串类型:提交表单的name和方法参数的名称是相同的,区分大小写。
<form action="account/form">
账户名:<input type="text" name="name"/>
金额:<input type="text" name="money"/>
<input type="submit" value="提交"/>
</form>
@RequestMapping("form")
public String getForm(String name,Integer money){
System.out.println(name+"---"+money);
return "success";
}
实体类型(JavaBean):提交表单的name和JavaBean中的属性名称需要一致,如果一个JavaBean类中包含其他的引用类型,那么表单的name属性需要编写为 对象.属性 例如:user.uname。
<form action="account/form">
账户名:<input type="text" name="name"/>
金额:<input type="text" name="money"/>
户主姓名:<input type="text" name="user.uname"/>
年龄:<input type="text" name="user.age"/>
<input type="submit" value="提交"/>
</form>
@RequestMapping("form")
public String getForm(Account account){
System.out.println(account);
System.out.println(account.getUser());
return "success";
}
public class Account implements Serializable {
private String name;
private Integer money;
private User user;
public void setter/getter/toString(){};
}
集合属性数据封装 JSP页面编写格式为 list[0].属性 、map[0].属性。
<form action="account/form">
List第1个元素名字:<input type="text" name="users[0].uname"/>
List第1个元素金额:<input type="text" name="users[0].age"/><br>
List第2个元素名字:<input type="text" name="users[1].uname"/>
List第2个元素金额:<input type="text" name="users[1].age"/><br>
Map第1个元素名字:<input type="text" name="userMap['one'].uname"/>
Map第1个元素金额:<input type="text" name="userMap['one'].age"/><br>
Map第2个元素名字:<input type="text" name="userMap['two'].uname"/>
Map第2个元素金额:<input type="text" name="userMap['two'].age"/><br>
<input type="submit"/>
</form>
@RequestMapping("form")
public String getForm(Account account){
System.out.println(account);
List<User> users = account.getUsers();
System.out.println(users);
Map<String, User> userMap = account.getUserMap();
Set<String> set = userMap.keySet();
for (String s : set) {
System.out.println(s+"---"+userMap.get(s));
}
return "success";
}
public class Account implements Serializable {
private String name;
private Integer money;
private User user;
private List<User> users;
private Map<String,User> userMap;
public void setter/getter/toString(){};
}
2. 请求参数中文乱码的解决
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>
<!-- 用来设置是否理会request.getCharacterEncoding()方法,设置为true则强制覆盖之前的编码格式 -->
<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 请求方式若出现编码问题,由于tomcat对 GET 和 POST 请求处理方式是不同的,则需要改 tomcat 的 server.xml配置文件。
在tomcat的server.xml中找到Connector标签。
<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 请求仍然乱码,则改成如下:
<Connector connectionTimeout="20000" port="8080"
protocol="HTTP/1.1" redirectPort="8443"
URIEncoding="UTF-8"/>
3. 自定义类型转换器
表单提交的任何数据类型全部都是字符串类型,但是后台定义Integer类型,数据也可以封装上,说明Spring框架内部会默认进行数据类型转换。
当你输入的日期格式为"2022/1/18"时,框架提供的默认转换器会自动帮你转换成Date类型,但是"2022-1-18"格式时,就会报错,这时候需要我们自己用 Convertor 接口编写一个自定义类型转换器。
切记我们要选择导入第二个带有两个泛型的Converter包,选第一个的话将使用不了。
public class MyConverter implements Converter<String,Date> {
@Override
public Date convert(String source) {
if(source==null){
throw new RuntimeException("空的。。。");
}
DateFormat df = new SimpleDateFormat("yyyy-MM-dd");
try {
return df.parse(source);
} catch (ParseException e) {
throw new RuntimeException("类型转换出错。。。");
}
}
}
下一步我们要在springmvc.xml中编写配置,来注册这个自定义类型转换器。
spring 配置类型转换器的机制是,将自定义的转换器注册到类型转换服务中去,所以我们要先创建一个转换服务conversionService进容器,再把我们的自定义转换器注入到这个工厂里去。
<!-- 注册自定义类型转换器 -->
<bean id="conversionService" class="org.springframework.context.support.ConversionServiceFactoryBean">
<property name="converters">
<set>
<bean id="myConverter" class="swz.MyConverter"/>
</set>
</property>
</bean>
<!-- 在 annotation-driven 标签中引用配置的类型转换服务 -->
<mvc:annotation-driven conversion-service="conversionService"></mvc:annotation-driven>
在上节基础配置中我们可以知道,annotation-driven 标签自动为我们配置了注解处理器和适配器,但其他一些服务,我们需要手动在这个标签中添加,比如上面的类型转换服务。这表明了annotation-driven 标签除了开启注解开发外,还有许多重要的功能。
4. 在控制器中使用原生的ServletAPI对象
想使用 Java Web 中的 老ServletAPI?很简单,只需要在控制器的方法参数定义HttpServletRequest和HttpServletResponse对象就可以用了。
@RequestMapping("test")
public String testServlet(HttpServletRequest request, HttpServletResponse response){
HttpSession session = request.getSession();
ServletContext servletContext = session.getServletContext();
System.out.println(request);
System.out.println(response);
System.out.println(session);
System.out.println(servletContext);
return "success";
}
Java Web 知识回顾
request.getSession().getServletContext() :获取的是page的上下文。
request.getSession().getServletContext().getRealPath(""):获取的的tamcat的路径,部署项目后相当于项目的路径。