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 接口编写一个自定义类型转换器。

spring boot 对于入参绑定对象的敏感字段 怎样处理_类型转换

切记我们要选择导入第二个带有两个泛型的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";
    }

spring boot 对于入参绑定对象的敏感字段 怎样处理_System_02

Java Web 知识回顾

request.getSession().getServletContext() :获取的是page的上下文。

request.getSession().getServletContext().getRealPath(""):获取的的tamcat的路径,部署项目后相当于项目的路径。