SpringMVC基础

SpringMVC基础

1.SpringMVC概述

1.1 三层架构

三层架构:

  • 表现层:负责数据展示
  • 业务层:负责业务处理
  • 数据层:负责数据操作

image-20210429150507447

1.2 MVC

MVC(Model View Controller),一种用于设计创建web应用程序表现层的模式

  • Model(模型):数据模型,用户封装数据
  • View(视图):页面视图,用户展示数据
    • jsp
    • html
  • Controller(控制器):处理用户交互的调度器,用于根据用户需求处理程序逻辑
    • Servlet
    • SpringMVC

image-20210429150913422

1.3 SpringMVC简介

概念:SpringMVC是一种基于Java实现MVC模型的轻量级web框架

SpringMVC优点:

  • 使用简单
  • 性能突出(相比现有的框架技术)
  • 灵活性强

2.入门案例

2.1 入门案例制作(重点

  • XML版

  • XML+注解版(主体

  • 纯注解版(变形

  • 基于Spring环境开发

  • SpringMVC版本与Spring版本同步–5.1.9

image-20210429170503625

步骤:

  1. 导入坐标
 <dependencies>
   <!--servlet3.1规范坐标-->
   <dependency>
     <groupId>javax.servlet</groupId>
     <artifactId>javax.servlet-api</artifactId>
     <version>3.1.0</version>
     <scope>provided</scope>
   </dependency>
 
   <!--jsp坐标-->
   <dependency>
     <groupId>javax.servlet.jsp</groupId>
     <artifactId>jsp-api</artifactId>
     <version>2.1</version>
     <scope>provided</scope>
   </dependency>
 
   <!--spring坐标-->
   <dependency>
     <groupId>org.springframework</groupId>
     <artifactId>spring-context</artifactId>
     <version>5.1.9.RELEASE</version>
     <scope>provided</scope>
   </dependency>
 
   <!--springmvc坐标-->
   <dependency>
     <groupId>org.springframework</groupId>
     <artifactId>spring-webmvc</artifactId>
     <version>5.1.9.RELEASE</version>
     <scope>provided</scope>
   </dependency>
 
   <!--spring web坐标-->
   <dependency>
     <groupId>org.springframework</groupId>
     <artifactId>spring-web</artifactId>
     <version>5.1.9.RELEASE</version>
     <scope>provided</scope>
   </dependency>
 </dependencies>
  <!--构建-->
  <build>
    <!--设置插件-->
    <plugins>
      <!--具体的插件配置-->
      <plugin>
        <groupId>org.apache.tomcat.maven</groupId>
        <artifactId>tomcat7-maven-plugin</artifactId>
        <version>2.1</version>
        <configuration>
          <port>80</port>
          <path>/</path>
        </configuration>
      </plugin>
    </plugins>
  </build>
  1. 定义业务层处理器Controller,并配置成spring的bean(等同于Servlet)

    @Controller
    public class UserController {
        @RequestMapping("/save")
        public String save() {
            System.out.println("user mvc controller is running...");
            return "success.jsp";
        }
    }
    

    该bean的处理需要使用独立的配置文件扫描(XML版):spring-mvc.xml

    <?xml version="1.0" encoding="UTF-8"?>
    <beans xmlns="http://www.springframework.org/schema/beans"
           xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
           xmlns:p="http://www.springframework.org/schema/p"
           xmlns:context="http://www.springframework.org/schema/context"
           xsi:schemaLocation="http://www.springframework.org/schema/beans
            https://www.springframework.org/schema/beans/spring-beans.xsd
            http://www.springframework.org/schema/context
            https://www.springframework.org/schema/context/spring-context.xsd">
        <!--文件扫描,配置作用范围-->
        <context:component-scan base-package="com.itheima"/>
    </beans>
    
  2. web.xml中配置SpringMVC核心控制器,用于将请求转发到对应的具体业务处理器Controller中(等同于Servlet配置)

    <servlet>
        <servlet-name>DispatcherServlet</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <init-param>
            <param-name>contextConfigLocaltion</param-name>
            <param-value>classpath*:spring-mvc.xml</param-value>
        </init-param>
    </servlet>
    <servlet-mapping>
        <servlet-name>DispatcherServlet</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>
    
  3. 设定具体Controller的访问路劲(等同于Servlet在web.xml中的配置),并设置返回页面

    @Controller
    public class UserController {
    
        @RequestMapping("/save")
        public String save() {
            System.out.println("user mvc controller is running...");
            return "success.jsp";
        }
    }
    

此处记录一个问题:

问题现象:org.apache.catalina.core.ContainerBase.addChildInternal ContainerBase.addChild: start: org.apache.catalina…

问题解决:

  1. 查看web.xml是否无误,路径等是否有误,一个servlet不能同时使用xml和注解配置,否则也很会出现该错误;

  2. 查看项目lib目录是否存在,idea默认不创建,

    进入Project Structures(ctrl+shift+alt+S)–>点击左侧Project Settings下的Artifacts。

    点击中间栏项目,这里会两个文件,一个是:项目名:war(war压缩包),一个是项目名:war exploded(war未压缩包)。

    点击war exploded项目,在右侧中第一栏Output Layout(项目发布生成的文件)下,展开WEB-INF文件夹,此时该目录下只有classes目录,无lib目录,自己手动创建一个lib目录,并点击上面+选择Library File添加maven导入的jar包

    重启tomcat就解决了(我是这样解决的,不保证所有类似问题都能解决)

2.2 入门案例工作流程分析(重点

服务器启动:

  1. 加载web.xml中的DispatcherServlet
  2. 读取spring-mvc.xml中的配置,加载所有com.itheima包中所有标记为bean的类
  3. 读取bean中方法上标注@RequestMapping("/save")的内容

处理请求:

  1. DispatcherServlet配置拦截所有请求 /
  2. 使用请求路径与所有加载的@RequestMapping的内容进行比对
  3. 执行对应的方法
  4. 根据方法的返回值在webapp目录中查找对应的页面并展示

2.3 SpringMVC技术架构图(重点

SpringMVC技术架构图:

image-20210429185141784

  • DispatcherServlet:前端控制器,是整体流程控制中心,由其调用其他组件处理用户的请求,有效的降低了组件间的耦合性
  • HandleMapping:处理器映射器,负责根据用户请求找到对应具体的Handler处理器
  • Handler:处理器,业务处理的核心类,通常由开发者编写,描述具体的业务
  • HandlerAdapter:处理器适配器,通过它对处理器进行执行
  • ViewResolver:视图解析器,将处理结果生成View视图
  • View:视图,最终产出结果,常用视图入jsp、html

3.基础配置

3.1 Controller加载控制(重点

SpringMVC的处理器对应bean必须按照规范格式开发,为避免加入无效的bean可通过bean加载过滤器进行包含设定或排除设定,表现层bean标注通常设为@Controller

<context:component-scan base-package="com.itheima">
    <context:include-filter
        type="annotation"
        expression="org.springframework.stereotype.Controller"/>
</context:component-scan>

以前使用方式:

@ComponentScan(
        value = "com.itheima",
        excludeFilters = 
                @ComponentScan.Filter(
                        type = FilterType.ANNOTATION,
                        classes = Controller.class)
)

此处记录一个问题:

问题现象:启动tomcat完成时,idea弹窗提示“http://localhost:80/找不到应用程序”

解决方法:点击idea主界面运行图标旁下拉列表,选择Edit Configuration,进入配置界面,Open Browser选择一个固定浏览器,不要使用默认的浏览器(tomcat启动完成自动使用该浏览器访问)

制作案例:和前面案例一样,只需修改spring-mvc.xml配置文件:

<context:component-scan base-package="com.itheima">
    <!--添加过滤,包含该注解才会被扫描到-->
    <context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
</context:component-scan>

bean加载控制说明:

  • 业务层与数据层bean加载由spring控制,参照spring课程加载方式
  • 表现层bean加载由SpringMVC单独控制
    • 表现层处理器bean使用注解@Controller声明
    • bean加载控制使用包含性过滤器
    • 过滤器类型为通过注解进行过滤
    • 过滤的注解名称为Controller

3.2 静态资源加载

核心控制器拦截的是所有请求,需要对非普通资源请求进行放行,通过配置放行资源实现

  • <mvc:resources mapping="/img/**" location="/img/"/>
    <mvc:resources mapping="/js/**" location="/img/"/>
    <mvc:resources mapping="/css/**" location="/img/"/>
    

使用简化格式可以放行所有普通资源调用,无需一一枚举

  • <mvc:default-servlet-handler/>
    

还是之前案例内容,只需做如下修改:

  1. 在webap下新建img目录,并导入一张图片

  2. 修改spring-mvc.xml配置文件:

    <?xml version="1.0" encoding="UTF-8"?>
    <beans xmlns="http://www.springframework.org/schema/beans"
           xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
           xmlns:context="http://www.springframework.org/schema/context"
           xmlns:mvc="http://www.springframework.org/schema/mvc"
           xsi:schemaLocation="http://www.springframework.org/schema/beans
            https://www.springframework.org/schema/beans/spring-beans.xsd
            http://www.springframework.org/schema/context
            https://www.springframework.org/schema/context/spring-context.xsd
            http://www.springframework.org/schema/mvc
            https://www.springframework.org/schema/mvc/spring-mvc.xsd">
    
        <context:component-scan base-package="com.itheima">
            <context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
        </context:component-scan>
    
    <!--
        <mvc:resources mapping="/img/**" location="/img/"/>
        <mvc:resources mapping="/js/**" location="/img/"/>
        <mvc:resources mapping="/css/**" location="/img/"/>
    -->
        <mvc:default-servlet-handler/>
    </beans>
    

注意:webapp下img目录需设置为resources目录,否则无法访问

3.3 中文乱码处理

SpringMVC提供专用的中文字符过滤器,用于处理乱码问题。在web.xml进行如下配置:

<?xml version="1.0" encoding="UTF-8"?>
<web-app
        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
        xmlns="http://java.sun.com/xml/ns/j2ee"
        xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee
         http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd"
        version="2.4">

  <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>
  </filter>
  <filter-mapping>
    <filter-name>CharacterEncodingFilter</filter-name>
    <url-pattern>/*</url-pattern>
  </filter-mapping>

  <servlet>
    <servlet-name>DispatcherServlet</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
    <init-param>
      <param-name>contextConfigLocation</param-name>
      <param-value>classpath*:spring-mvc.xml</param-value>
    </init-param>
  </servlet>
  <servlet-mapping>
    <servlet-name>DispatcherServlet</servlet-name>
    <url-pattern>/</url-pattern>
  </servlet-mapping>
</web-app>  

3.4 注解驱动

使用注解形式转化SpringMVC核心配置文件为配置类

@Configuration
@ComponentScan(value = "com.itheima",
        excludeFilters = @ComponentScan.Filter(
                type = FilterType.ANNOTATION,
                classes = Controller.class)
        )
public class SpringMvcConfiguration {
}

操作过程:

  1. 创建配置类

    @Configuration
    @ComponentScan(value = "com.itheima",includeFilters =
        @ComponentScan.Filter(type = FilterType.ANNOTATION,classes = {Controller.class})
    )
    public class SpringMVCConfiguration implements WebMvcConfigurer {
    //    @Override
    //    public void addResourceHandlers(ResourceHandlerRegistry registry) {
    //        registry.addResourceHandler("/img/**").addResourceLocations("/img/");
    //        registry.addResourceHandler("/js/**").addResourceLocations("/js/");
    //        registry.addResourceHandler("/css/**").addResourceLocations("/css/");
    //    }
    
        @Override
        public void configureDefaultServletHandling(DefaultServletHandlerConfigurer configurer) {
            configurer.enable();
        }
    }
    
  2. 基于servlet3.0规范,自定义servlet容器初始化配置类,加载SpringMVC核心配置类

    public class ServletContainersInitConfig extends AbstractDispatcherServletInitializer {
    
        @Override
        protected WebApplicationContext createServletApplicationContext() {
            AnnotationConfigWebApplicationContext ctx = new AnnotationConfigWebApplicationContext();
            ctx.register(SpringMVCConfiguration.class);
            return ctx;
        }
    
        @Override
        protected String[] getServletMappings() {
            return new String[]{"/"};
        }
    
        @Override
        protected WebApplicationContext createRootApplicationContext() {
            return null;
        }
    
        @Override
        public void onStartup(ServletContext servletContext) throws ServletException {
            super.onStartup(servletContext);
            CharacterEncodingFilter cef = new CharacterEncodingFilter();
            cef.setEncoding("utf-8");
            FilterRegistration.Dynamic registration = servletContext.addFilter("characterEncodingFilter", cef);
            registration.addMappingForUrlPatterns(EnumSet.of(DispatcherType.REQUEST,DispatcherType.FORWARD,DispatcherType.INCLUDE),false,"/*");
        }
    }
    
  3. 静态资源加载过滤(注解版)

    配置类实现WebMvcConfigurer接口,覆盖addResourceHandlers方法,在其中对具体的资源进行设定

    @Override
    public void addResourceHandlers(ResourceHandlerRegistry registry) {
        registry.addResourceHandler("/img/**").addResourceLocations("/img/");
        registry.addResourceHandler("/js/**").addResourceLocations("/js/");
        registry.addResourceHandler("/css/**").addResourceLocations("/css/");
    }
    

    或者覆盖configureDefaultServletHandling方法,使用Servlet默认过滤规则

    @Override
    public void configureDefaultServletHandling(DefaultServletHandlerConfigurer configurer) {
        configurer.enable();
    }
    
  4. 中文乱码处理(注解版)

    Servlet3.0规范启动服务器时做的工作通过实现ServletContainerInitializer接口,在onStartup方法中完成,包括监听器注册、过滤器注册等

    @Override
    public void onStartup(ServletContext servletContext) throws ServletException {
        //调用父类onStartup方法
        super.onStartup(servletContext);
        //1.创建字符集过滤器对象
        CharacterEncodingFilter cef = new CharacterEncodingFilter();
        //2.设置使用的字符集
        cef.setEncoding("utf-8");
        //3.添加到容器(它不是ioc容器,而是ServletContainer)
        FilterRegistration.Dynamic registration = servletContext.addFilter("characterEncodingFilter", cef);
        //4.添加映射
        registration.addMappingForUrlPatterns(EnumSet.of(DispatcherType.REQUEST,DispatcherType.FORWARD,DispatcherType.INCLUDE),false,"/*");
    }
    

4.请求

4.1 请求参数(重点

SpringMVC将传递的参数封装到处理器方法的形参中,达到快速访问参数的目的

访问URL:http://localhost/requestParam?name=itheima

@RequestMapping("/requestParam")
public String requestParam(String name) {
    System.out.println("name = "+name);
    return "page.jsp";
}

请求参数类型:

  • 普通类型参数
  • POJO类型参数
  • 数组类型参数
  • 集合类型参数

普通类型参数

  • 参数名与处理器方法形参名保持一致

  • 访问URL:http://localhost/requestParam?name=itheima&age=20

  •  @RequestMapping("/requestParam1")
     public String requestParam1(String name,String age) {
         System.out.println("name = "+name+", age = "+age);
         return "page.jsp";
     }
    

参数设定

名称:@RequestParam

类型:形参注解

位置:处理器类中的方法形参前方

作用:绑定请求参数与对应处理方法形参间的关系

范例:

  • @RequestMapping("/requestParam2")
    public String requestParam2(@RequestParam(value = "userName",required = true,defaultValue = "value") String name) {
        System.out.println("name = "+name);
        return "page.jsp";
    }
    

    注意:若设置了required = true,则在 浏览器URL地址栏必须加上参数,否则报错404。可以通过设置默认参数defaultValue = "value"避免未输入报错

    访问路径:http://localhost/requestParam2?userName=zhangsan

POJO类型参数

  • 当POJO中使用简单类型属性时,参数名称与POJO类属性名保持一致

  • 访问URL:http://localhost/requestParam?name=itheima&age=20

  • public class User {
        private String name;
        private Integer age;
        //省略getter、setter和toString方法,下同
    }    
    
  • @RequestMapping("/requestParam3")
    public String requestParam3(User user) {
        System.out.println("user = "+user);
        return "page.jsp";
    }
    

    注意:这里需要注释掉上面的方法requestParam2,否者此处也需要给参数添加注解@RequestParam,否则报错500:java.lang.IllegalArgumentException: Name for argument type [java.lang.String] not available, and par…

参数冲突

  • 当POJO类型属性与其他参数出现同名问题时,将被同时覆盖

  • 访问URL:http://localhost/requestParam4?name=Jock&age=30

  • @RequestMapping("/requestParam4")
    public String requestParam4(User user,String name) {
        System.out.println("user = "+user+", name = "+name);
        return "page.jsp";
    }
    

    控制台打印结果:user = User{name=‘Jock’, age=30}, name = Jock

  • 建议使用@RequestParam注解进行区分

  • 注意:这里也需要注释上一个方法,否者报错500:java.lang.IllegalStateException: Method [requestParam4] was discovered in the .class file but cannot be resolved in the class object

复杂POJO类型参数

  • 当POJO中出现对象属性时,参数名称与对象层次结构名称保持一致

  • 访问URL:http://localhost/requestParam5?address.city=bejing

  • public class Address {
        private String province;
        private String city;
        private String address;
    }
    
    public class User {
        private String name;
        private Integer age;
        private Address address;
    }
    
    @RequestMapping("/requestParam5")
    public String requestParam5(User user) {
        System.out.println("city = "+user.getAddress().getCity());
        return "page.jsp";
    }
    

    控制台打印结果:city = bejing

  • 当POJO中出现集合,保存简单数据,使用多个相同名称的参数为其进行赋值

  • 访问URL:http://localhost/requestParam6?nick=Jock1&nick=Jock2&nick=Jocker

  • public class User {
        private String name;
        private Integer age;
    
        private Address address;
        private List<String> nick;
    }
    
    @RequestMapping("/requestParam6")
    public String requestParam6(User user) {
        System.out.println("user = "+user);
        return "page.jsp";
    }
    

    控制台打印结果:user = User{name=‘null’, age=null, address=null, nick=[Jock1, Jock2, Jocker]}

  • 当POJO中出现List,保存对象数据,参数名称与对象层次结构名称保持一致,使用数组格式描述集合中对象的位置

  • 访问URL:http://localhost/requestParam7?addresses[0].city=beijing&addresses[1].province=hebei

  • public class User {
        private String name;
        private Integer age;
    
        private Address address;
        private List<String> nick;
        private List<Address> addresses;
    }
    
    @RequestMapping("/requestParam7")
    public String requestParam7(User user) {
        System.out.println("addresses = "+user.getAddresses());
        return "page.jsp";
    }
    

    访问报错400:java.lang.IllegalArgumentException: 在请求目标中找到无效字符。有效字符在RFC 7230和RFC 3986中定义

    百度解决方案:tomcat配置文件–>conf–>server.xml,修改内容

    <Connector port="8080" protocol="HTTP/1.1" connectionTimeout="20000" redirectPort="8443" URIEncoding="utf-8"
    		   relaxedPathChars="|{}[],%"
    		   relaxedQueryChars="|{}[],%" />
    

    然而,修改了还是报该错误

  • 当POJO中出现Map,保存对象数据,参数名称与对象层次结构名称保持一致,使用映射格式描述集合对象的位置

  • 访问URL:http://localhost/requestParam8?addressMap[‘home’].city=beijing&addressMap[‘job’].province=hebei

  • public class User {
        private String name;
        private Integer age;
    
        private Address address;
        private List<String> nick;
        private List<Address> addresses;
        private Map<String,Address> addressMap;
    }
    
    @RequestMapping("/requestParam8")
    public String requestParam8(User user) {
        System.out.println("addressMap = "+user.getAddressMap());
        return "page.jsp";
    }
    

    同样报错

数组类型参数

  • 请求参数名与处理器方法形参名保持一致,且请求参数数量>1个

  • 访问URL:http://localhost/requestParam9?nicks=jock&nicks=jocker

  • @RequestMapping("/requestParam9")
    public String requestParam9(String[] nicks) {
        for (String nick : nicks) {
            System.out.println("name = "+nick);
        }
        return "page.jsp";
    }
    
  • 控制台打印结果:name = jock name = jocker

集合类型参数

  • 保存简单类型数据,请求 参数名与处理器方法形参名保持一致,且请求参数数量>1个

  • 访问URL:http://localhost/requestParam10?nick=jock&nick=jocker

  • @RequestMapping("/requestParam10")
    public String requestParam10(@RequestParam("nick") List<String> nick) {
        System.out.println("nick = "+nick);
        return "page.jsp";
    }
    
  • 控制台打印结果:nick = [jock, jocker]

  • 注意:SpringMVC默认将List作为对象处理,赋值前先创建对象,然后将nick作为对象的属性进行处理。由于List是接口,无法创建对象,报错无法找到构造方法异常;修复类型可创建对象的ArrayList类型后,对象可以创建,但没有nick属性,因此数据为空。此时需要告知SpringMVC的处理器nick是一组数据,而不是一个单一数据。通过@RequestParam注解,将数量大于1个names参数打包成参数数组后,SpringMVC才能识别该数据格式,并判定形参类型是否为数组或集合,并按照数组或集合对象的形式操作数据。

4.2 类型转换器

SpringMVC对接收的数据进行自动类型转换,该工作通过Converter接口实现

image-20210430175046003

标量转换器:

  • StringToBooleanConverter String → Boolean
  • ObjectToStringConverter Object → String
  • StringToNumberConverterFactory String → Number(Integer、Long等)
  • NumberToNumberConverterFactory Number子类型之间(Integer、Long、Double等)
  • StringToCharsetConverter String → java.lang.Character
  • NumberToCharacterConverter Number子类型(Integer、Long、Double等) → java.lang.Character
  • CharacterToNumberFactory java.lang.Character → Number子类型(Integer、Long、Double等)
  • StringToEnumConverterFactory String → Enum类型
  • EnumToStringConverter Enum类型 → String
  • StringToLocaleConverter String → java.util.Local
  • PropertiesToStringConverter java.util.Properties → String
  • StringToPropertiesConverter String → java.util.Properties

集合、数组相关转换器

  • ArrayToCollectionConverter 数组 → 集合(List、Set)

默认转换器

  • ObjectToObjectConverter Object间

  • IdToEntityConverter Id → Entity

  • FallbackObjectToStringConverter Object → String

  • 访问URL:http://localhost/requestParam11?date=2021/4/30

  • @RequestMapping("/requestParam11")
    public String requestParam11(Date date) {
        System.out.println("date = "+date);
        return "page.jsp";
    }
    

    控制台打印结果:date = Fri Apr 30 00:00:00 CST 2021

  • 当输入格式变换会报错,如:http://localhost/requestParam11?date=2021-4-30

日期类型格式转换

  • 修改spring-mvc.xml配置使输入格式满足需求

  • <!--6.启用自定义converter-->
    <mvc:annotation-driven conversion-service="conversionService"/>
    <!--1.设定格式类型Converter,注册为Bean,受SpringMVC管理-->
    <bean id="conversionService" class="org.springframework.format.support.FormattingConversionServiceFactoryBean">
        <!--2.自定义Converter格式类型设定,该设定使用的是同类型覆盖的思想-->
        <property name="formatters">
            <!--3.使用set保障同类型的转换器仅保留一个,避免冲突-->
            <set>
                <!--4.设置具体的格式类型-->
                <bean class="org.springframework.format.datetime.DateFormatter">
                    <!--5.类型规则-->
                    <property name="pattern" value="yyyy-MM-dd"/>
                </bean>
            </set>
        </property>
    </bean>
    

日期类型格式转换(简化版)

  • 名称:DateTimeFormat

  • 类型:形参注解、成员变量注解

  • 位置:形参前面 或 成员变量上方

  • 作用:为当前参数或变量指定类型转换规则

  • 范例:

  • @RequestMapping("/requestParam12")
    public String requestParam12(@DateTimeFormat(pattern = "yyyy-MM-dd") Date date) {
        System.out.println("date = "+date);
        return "page.jsp";
    }
    
    @DateTimeFormat(pattern = "yyyy-MM-dd")
    private Date birthday;
    

    注意:依赖注解驱动支持

    <mvc:annotation-driven />
    

自定义类型转换器

  • 自定义类型转换器,实现Converter接口,并指定转换前与转换后的类型

  • public class MyDateConverter implements Converter<String, Date> {
        public Date convert(String source) {
            DateFormat df = new SimpleDateFormat("yyyy-MM-dd");
            Date date = null;
            try {
                date = df.parse(source);
            } catch (ParseException e) {
                e.printStackTrace();
            }
            return date;
        }
    }
    

    注册自定义转换器

    <mvc:annotation-driven conversion-service="conversionService"/>
    
    <bean id="conversionService" class="org.springframework.context.support.ConversionServiceFactoryBean">
        <property name="converters">
            <set>
                <bean class="com.itheima.converter.MyDateConverter"></bean>
            </set>
        </property>
    </bean>
    

    通过注册自定义转换器,将该功能加入到SpringMVC的转换服务ConverterService中

4.3 请求映射(重点

名称:@RequestMapping

类型:方法注解

位置:处理器类中方法定义上方

作用:绑定请求地址与对应处理方法间的关系

范例:

  • @RequestMapping("/requestURL")
    public String requestParam12(@DateTimeFormat(pattern = "yyyy-MM-dd") Date date) {
        System.out.println("date = "+date);
        return "page.jsp";
    }
    
  • 访问路径:/requestURL

  • 请求返回的页面地址默认为当前路径

  • 当设置了公共的访问前缀后,当前路径发生了变化,需要根据变化修改地址或修改访问页面的路径

  • 访问root路径下的页面

    @Controller
    public class UserController {
        @RequestMapping("/requestURL1")
        public String requestURL1() {
            System.out.println();
            return "page.jsp";
        }
    }
    
  • 访问root路径下的user路径下的页面

    @Controller
    @RequestMapping("/user")
    public class UserController {
            @RequestMapping("/requestURL2")
        public String requestURL2() {
            System.out.println();
            return "/page.jsp";
        }
    }
    

@RequestMapping属性

  • 常用属性

    • value
    • method
    @RequestMapping(
            value = "/requestURL3",     //设定请求路径,与path、name属性相同
            params = "name",            //设定请求参数
            method = RequestMethod.GET, //设定请求方式
            headers = "content-type=text/*",//设定请求消息头
            consumes = "text/*",        //用于指定可以接收的请求正文类型(MIME类型)
            produces = "text/*"         //用于指定可以生成的响应正文类型(MIME类型)
    )
    public String requestURL3() {
        System.out.println();
        return "page.jsp";
    }
    

5.响应

5.1 无数据跳转页面

响应方式:(数据流)

  • 页面
    • HTML(页面)
    • JSP(页面+数据)
  • 数据
    • JSON数据
    • XML数据
    • 文本数据
  • 文件
    • 数据流

页面跳转设定:

  • 当处理器方法的返回值类型为String类型,即可通过具体的返回字设置访问的页面

    @Controller
    public class UserController {
        @RequestMapping("/showPage")
        public String showPage() {
            System.out.println("user mvc controller is running...");
            return "page.jsp";
        }
    }    
    

页面跳转方式:

  • 转发(默认)

    @RequestMapping("/showPage1")
    public String showPage1() {
        System.out.println("user mvc controller is running...");
        return "forward:page.jsp";
    }
    
  • 重定向

    @RequestMapping("/showPage2")
    public String showPage2() {
        System.out.println("user mvc controller is running...");
        return "redirect:page.jsp";
    }
    

    注意:页面访问地址所携带的/可以使用。但是在WEB-INF目录下的资源forward可以访问,而redirect不能访问。

页面访问快捷设定

  • 展示页面的保存位置通常固定,且结构相似,可以设定通用的访问路径,简化页面配置格式(spring-mvc.xml)

    <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
        <property name="prefix" value="/WEB-INF/page/"/>
        <property name="suffix" value=".jsp"/>
    </bean>
    
    @RequestMapping("/showPage3")
    public String showPage3() {
        System.out.println("user mvc controller is running...");
        return "page";
    }
    
  • 使用了简化配置,跳转页面名称前不能加forward,否则报错

    @RequestMapping("/showPage4")
    public String showPage4() {
        System.out.println("user mvc controller is running...");
        return "forward:page";//HTTP状态 404 - 未找到
    }
    

页面访问快捷设定缺省页面:

  • 如果设定了返回值,使用void类型,则默认使用访问路径作页面地址的前缀后缀

    @RequestMapping("/showPage5")
    public void showPage5() {
    }
    

    等同于

    @RequestMapping("/showPage5")
    public String showPage5() {
        return "showPage5";
    }
    

5.2 带数据跳转页面(重点

  • 方式一:使用HttpServletRequest类型形参进行数据传递

    @RequestMapping("showPageAndData1")
    public String showPageAndData1(HttpServletRequest request) {
        request.setAttribute("name","itheima");
        return "page";
    }
    
  • 方式二:使用Model类型形参进行数据传递

    @RequestMapping("showPageAndData2")
    public String showPageAndData2(Model model) {
        model.addAttribute("name","Jock");
        Book book = new Book();
        book.setName("SpringMVC入门案例");
        book.setPrice(66.66);
        model.addAttribute(book);
        return "page";
    }
    
  • 方式三:使用ModelAndView类型形参进行数据传递,将该对象作为返回字传递给调用者

        @RequestMapping("showPageAndData3")
        public ModelAndView showPageAndData3(ModelAndView modelAndView) {
    //        ModelAndView mv = new ModelAndView(); //用于替换形参中的ModelAndView
            Book book = new Book();
            book.setName("SpringMVC入门案例");
            book.setPrice(66.66);
    
            modelAndView.addObject("book",book);
            modelAndView.addObject("name","Jocker");
    
            modelAndView.setViewName("page");//等同于return "page"
    
            return modelAndView;
        }
    

    重定向:

    @RequestMapping("showPageAndData5")
    public ModelAndView showPageAndData5(ModelAndView modelAndView) {
        modelAndView.setViewName("redirect:page");//等同于return "redirect:page.jsp";
        return modelAndView;
    }
    

    注意:使用重定向需要注释前面设置的spring-mvc.xml配置中添加的WEB-INF/page目录,因为WEB-INF有安全访问,重定向不能访问。

5.3 存数据返回(JSON)(重点

导入json坐标pom.xml

  <!--json相关坐标3个-->
  <dependency>
    <groupId>com.fasterxml.jackson.core</groupId>
    <artifactId>jackson-core</artifactId>
    <version>2.9.0</version>
  </dependency>

  <dependency>
    <groupId>com.fasterxml.jackson.core</groupId>
    <artifactId>jackson-databind</artifactId>
    <version>2.9.0</version>
  </dependency>

  <dependency>
    <groupId>com.fasterxml.jackson.core</groupId>
    <artifactId>jackson-annotations</artifactId>
    <version>2.9.0</version>
  </dependency>
</dependencies>

运行tomcat可能会报错:rg.apache.catalina.core.StandardContext.filterStart 启动过滤器异常
java.lang.ClassNotFoundException: org.springframework.web.filter.CharacterEncodingFilter

解决方案:在上面讲过,Project Structures(ctrl+shift+alt+S)–> Project Settings下的Artifact --> swar exploded项目 --> Output Layout --> 展开WEB-INF–>手动创建一个lib目录(由于上面已经创建了,此处不再创建),并点击上面+选择Library File添加maven导入的jar包。重启tomcat就解决了

返回数据:

  • 方式一:使用response对象完成数据返回

    @RequestMapping("/showData1")
    public void showData1(HttpServletResponse response) throws IOException {
        response.getWriter().write("message");
    }
    
  • 方式二:(简化格式)

    @RequestMapping("/showData2")
    @ResponseBody
    public String showData2() {
        return "message";
    }
    

返回JSON数据:

  • 方式一:基于response返回数据的简化格式,返回json数据

    @RequestMapping("/showData2")
    @ResponseBody
    public String showData2() {
        return "{'name':'Jock'}";
    }
    
    @RequestMapping("/showData3")
    @ResponseBody
    public String showData3() throws JsonProcessingException {
        Book book = new Book();
        book.setName("SpringMVC入门案例");
        book.setPrice(66.66);
    
        ObjectMapper om = new ObjectMapper();
        return om.writeValueAsString(book);
    }
    
  • 方式二(对象):使用SpringMVC提供的消息类型转换器将对象与集合数据自动转换为json数据

    @RequestMapping("/showData4")
    @ResponseBody
    public Book showData4() {
        Book book = new Book();
        book.setName("SpringMVC入门案例");
        book.setPrice(66.66);
        return book;
    }
    

    spring-mvc.xml添加注解驱动:<mvc:annotation-driven/>

  • 方式三(集合):使用SpringMVC注解驱动简化配置

    <mvc:annotation-driven/>
    
    @RequestMapping("/showData5")
    @ResponseBody
    public List<Book> showData5() {
        Book book1 = new Book();
        book1.setName("SpringMVC入门案例1");
        book1.setPrice(66.66);
    
        Book book2 = new Book();
        book2.setName("SpringMVC入门案例2");
        book2.setPrice(88.88);
    
        List<Book> list = new ArrayList<Book>();
        list.add(book1);
        list.add(book2);
    
        return list;
    }
    
  • 注解驱动格式

    @Configuration
    @ComponentScan("com.itheima")
    public class SpringMVCConfiguration {
        
    }
    

6.Servlet相关接口

6.1 HttpServletRequest

SpringMVC提供访问原始Servlet接口API的功能,通过形参声明即可

@RequestMapping("servletApi")
public String servletApi(HttpServletRequest request, HttpServletResponse response, HttpSession session) {
    System.out.println(request);
    System.out.println(response);
    System.out.println(session);
    return "page";
}

打印结果:

org.apache.catalina.connector.RequestFacade@1f787c3e
org.apache.catalina.connector.ResponseFacade@45b7d257
org.apache.catalina.session.StandardSessionFacade@411f594e

6.2 HttpServletResponse

6.3 HttpSession

6.4 Head

Head数据获取:

  • 名称:@RequestHeader

  • 类型:形参注解

  • 位置:处理器类中的方法形参前方

  • 作用:绑定请求头数据与对应处理方法形参间的关系

  • 范例:

    @RequestMapping("headApi")
    public String headApi(@RequestHeader("Accept-Language") String head ) {
        System.out.println(head);
        return "page";
    }
    

    打印结果:zh-CN,zh;q=0.9,en;q=0.8,en-GB;q=0.7,en-US;q=0.6

6.5 Cookie

Cookie数据获取:

  • 名称:@CookieValue

  • 类型:形参注解

  • 位置:处理器类中的方法形参前方

  • 作用:绑定请求Cookie数据与对应处理方法形参间的关系

  • 范例:

    @RequestMapping("cookieApi")
    public String cookieApi(@RequestHeader("JSESSIONID") String jsessionid ) {
        System.out.println(jsessionid);
        return "page";
    }
    

    打印结果:3AE4C46FC4AE80D62FAA29DA2DDD383F

6.6 Session

Session数据获取:

  • 名称:@SessionAttribute

  • 类型:形参注解

  • 位置:处理器类中的方法形参前方

  • 作用:绑定请求Session数据与对应处理方法形参间的关系

  • 范例:

    @RequestMapping("sessionApi")
    public String sessionApi(@SessionAttribute("name") String name) {
        System.out.println(name);
        return "page";
    }
    

    打印结果:itheima

Session数据设置(了解):

  • 名称:@SessionAttributes

  • 类型:类注解

  • 位置:处理器类上方

  • 作用:声明放入Session范围的变量名称,适用于Model类型数据传参

  • 范例:

    @Controller
    @SessionAttributes(names = {"age","gender"})
    public class UserController {
        //测试用方法,设置session数据
        @RequestMapping("setSessionData")
        public String setSessionData(HttpSession session) {
            session.setAttribute("name","itheima");
            return "page";
        }
    
    
        @RequestMapping("sessionApi")
        public String sessionApi(@SessionAttribute("name") String name,
                                 @SessionAttribute("age") int age,
                                 @SessionAttribute("gender") String gender) {
            System.out.println(name);
            System.out.println(age);
            System.out.println(gender);
            return "page";
        }
    
        //配合@SessionAttributes(names = {"age","gender"})使用
        @RequestMapping("setSessionData2")
        public String setSessionData2(Model model) {
            model.addAttribute("age","23");
            model.addAttribute("gender","男");
            return "page";
        }
    }    
    

    打印结果:itheima
    23

6.7 注解式参数数据封装底层原理

  • 数据的来源不同,对应的处理策略要进行区分
    • Head
    • Cookie
  • SpringMVC使用策略模式进行处理分发
    • 顶层接口:HandlerMethodArgumentResolver
    • 实现类:…

原文地址:https://blog.csdn.net/qq_53529114/article/details/116329507

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


开发过程中是不可避免地会出现各种异常情况的,例如网络连接异常、数据格式异常、空指针异常等等。异常的出现可能导致程序的运行出现问题,甚至直接导致程序崩溃。因此,在开发过程中,合理处理异常、避免异常产生、以及对异常进行有效的调试是非常重要的。 对于异常的处理,一般分为两种方式: 编程式异常处理:是指在代
说明:使用注解方式实现AOP切面。 什么是AOP? 面向切面编程,利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。 通俗描述:不通过修改源代码方式,在主干功能里面添加新功能。 AOP底层使用动态代理。 AOP术语 连接点
Spring MVC中的拦截器是一种可以在请求处理过程中对请求进行拦截和处理的机制。 拦截器可以用于执行一些公共的操作,例如日志记录、权限验证、数据转换等。在Spring MVC中,可以通过实现HandlerInterceptor接口来创建自定义的拦截器,并通过配置来指定拦截器的应用范围和顺序。 S
在 JavaWeb 中,共享域指的是在 Servlet 中存储数据,以便在同一 Web 应用程序的多个组件中进行共享和访问。常见的共享域有四种:ServletContext、HttpSession、HttpServletRequest、PageContext。 ServletContext 共享域:
文件上传 说明: 使用maven构建web工程。 使用Thymeleaf技术进行服务器页面渲染。 使用ResponseEntity实现下载文件的功能。 @Controller public class FileDownloadAndUpload { @GetMapping(&quot;/file/d
创建初始化类,替换web.xml 在Servlet3.0环境中,Web容器(Tomcat)会在类路径中查找实现javax.servlet.ServletContainerInitializer接口的类,如果找到的话就用它来配置Servlet容器。 Spring提供了这个接口的实现,名为SpringS
在 Web 应用的三层架构中,确保在表述层(Presentation Layer)对数据进行检查和校验是非常重要的。正确的数据校验可以确保业务逻辑层(Business Logic Layer)基于有效和合法的数据进行处理,同时将错误的数据隔离在业务逻辑层之外。这有助于提高系统的健壮性、安全性和可维护
什么是事务? 事务(Transaction)是数据库操作最基本单元,逻辑上一组操作,要么都成功,要么都失败,如果操作之间有一个失败所有操作都失败 。 事务四个特性(ACID) 原子性 一组操作要么都成功,要么都失败。 一致性 一组数据从事务1合法状态转为事务2的另一种合法状态,就是一致。 隔离性 事
什么是JdbcTemplate? Spring 框架对 JDBC 进行封装,使用 JdbcTemplate 方便实现对数据库操作。 准备工作 引入jdbcTemplate的相关依赖: 案例实操 创建jdbc.properties文件,配置数据库信息 jdbc.driver=com.mysql.cj.
SpringMVC1.MVC架构MVC是模型(Model)、视图(View)、控制器(Controller)的简写,是一种软件设计规范是将业务逻辑、数据、显示分离的方法来写代码MVC主要作用是:降低了视图和业务逻辑之间的双向耦合MVC是一个架构模型,不是一种设计模式。1.model(模型)数据模型,提供要展示的数据,因此包
SpringMVC学习笔记1.SpringMVC应用1.1SpringMVC简介​SpringMVC全名叫SpringWebMVC,是⼀种基于Java的实现MVC设计模型的请求驱动类型的轻量级Web框架,属于SpringFrameWork的后续产品。​MVC全名是ModelViewController,是模型(model)-视图(view)-控制器(co
11.1数据回显基本用法数据回显就是当用户数据提交失败时,自动填充好已经输入的数据。一般来说,如果使用Ajax来做数据提交,基本上是没有数据回显这个需求的,但是如果是通过表单做数据提交,那么数据回显就非常有必要了。11.1.1简单数据类型简单数据类型,实际上框架在这里没有
一、SpringMVC简介1、SpringMVC中重要组件DispatcherServlet:前端控制器,接收所有请求(如果配置/不包含jsp)HandlerMapping:解析请求格式的.判断希望要执行哪个具体的方法.HandlerAdapter:负责调用具体的方法.ViewResovler:视图解析器.解析结果,准备跳转到具体的物
1.它们主要负责的模块Spring主要应用于业务逻辑层。SpringMVC主要应用于表现层。MyBatis主要应用于持久层。2.它们的核心Spring有三大核心,分别是IOC(控制反转),DI(依赖注入)和AOP(面向切面编程)。SpringMVC的核心是DispatcherServlet(前端控制器)。MyBatis的核心是ORM(对
3.注解开发Springmvc1.使用注解开发要注意开启注解支持,2.注解简化了,处理映射器和处理适配器,只用去管视图解析器即可案例代码:1.web.xml,基本不变可以直接拿去用<!--调用DispatcherServlet--><servlet><servlet-name>springmvc</servlet-name>
拦截器概述SpringMVC的处理器拦截器类似于Servlet开发中的过滤器Filter,用于对处理器进行预处理和后处理。开发者可以自己定义一些拦截器来实现特定的功能。**过滤器与拦截器的区别:**拦截器是AOP思想的具体应用。过滤器servlet规范中的一部分,任何javaweb工程都可以使用
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:context="http://www.springframework.org/schema/context"xmlns:mvc="http://www.springframework.org/schema/mvc"xmlns:xsi="
学习内容:1、SSH&SSM2、Spring3、Struts2&SpringMVC4、Hibernate&MyBatis学习产出:1.SSH和SSM都是有Spring框架的,他们两个差不多。2.Spring分为四个模块,持久层,表示层,检测层,还有核心层,核心层分为2个关键核心功能。分别为,控制反转(IOC),依赖注入(DI),和面向切面编程
一、SpringMVC项目无法引入js,css的问题具体原因是css和js等被SpringMVC拦截了:解决方案:在spring-mvc.xml中配置<mvc:default-servlet-handler/><?xmlversion="1.0"encoding="UTF-8"?><beansxmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
开发环境:Eclipse/MyEclipse、Tomcat8、Jdk1.8数据库:MySQL前端:JavaScript、jQuery、bootstrap4、particles.js后端:maven、SpringMVC、MyBatis、ajax、mysql读写分离、mybatis分页适用于:课程设计,毕业设计,学习等等系统介绍