JAVA框架-SpringMVC1(配置和入门程序)

Spring入门

SpringMVC框架概述

Spring Web MVC是基于Servlet API构建的传统Web框架,并且从一开始就已包含在Spring框架中

与Spring Web MVC并行,Spring Framework 5.0引入了一个新的反应式Web框架,其名称“ Spring WebFlux;

理解:

首先SpringMVC 是一个MVC构架模式的web框架,是基于Servlet的,从Spring第一个版本就一起推出了,

传统web框架,指的是SpringMVC依然使用多线程同步并发的方式来处理请求,现如今大家都在鼓吹异步并发多么多么好,从测试数据来看异步并发效率的确更好,但是其并不成熟,极大多数公司项目还没有更新到异步技术,盲目的进行重构可能会引发更多的问题, 并且异步编程在代码结构上会产生较大的变化,对于初学者而言,掌握难度是较大的;

构架图:

SpringMVC在系统中的位置:

可以看出

SpringMVC 并没有代替Servlet,它只是在Servlet上提供了一套封装好的组件,提高开发效率;

还使得开发出的项目更加规范;否则每个人可能有每个人不同的MVC;

SpringMVC核心组件

思考:

若没有SpringMVC框架,我们该如何去编写一个较大的web项目呢,可以发现在选课系统中出现了大量的Servlet,因为一个请求地址就需要一个Servlet,使得项目体积变大,且Servlet是长期存在内存的;

第一步,我们希望用一个Servlet来处理多个请求甚至是所有请求,就需要实现能根据请求路径查找处理请求方法的逻辑,这也是SpringMVC要做的第一件事情;

  • DispatcherServlet:前端控制器
    用户请求首先到达前端控制器,它就相当于mvc模式中的c,DispatcherServlet是整个流程控制的调度中心,由它调用其它组件处理用户的请求,DispatcherServlet的存在降低了组件之间的耦合性。
  • Handler:处理器
    Handler是继DispatcherServlet前端控制器的后端控制器,DispatcherServlet会将请求发送至对应的Handler来进行处理。Handler是处理业务逻辑的地方,需要我们自己来编写具体代码,等同于之前的Service层
  • HandlerMapping:处理器映射器
    HandlerMapping负责根据用户请求路径找到Handler,springmvc提供了不同的映射器实现不同的映射方式,例如:BeanName映射,配置文件映射,注解映射等。
  • HandlAdapter:处理器适配器
    通过HandlerAdapter来执行Handler,因为Handler有不同形式,意味着调用方式是不同的,这是适配器模式的应用,我们也可以扩展适配器来实现新的Handler;
  • ViewResolver:视图解析器
    ViewResolver负责从Handler中获取数据和视图,根据逻辑视名称查找物理视图文件,并查找View对象,再生成View对象;
  • View:视图
    View的职责就是装配数据,SpringMVC框架提供了很多的View视图类型的支持,包括:jstlView、freemarkerView、pdfView等。常用视图就是jsp。我们需要根据业务需求,通过页面标签或页面模版技术将模型数据展示给用户

当然还有一些其他的

类型 说明
HandlerExceptionResolver Handler异常处理器
LocaleResolver 提供国际化的视图。根据不同地区显示不同内容
ThemeResolver 根据不同地区提供个性化的布局
MultipartResolver 解析multipart请求数据,如浏览器表单文件上传
FlashMapManager 常用于通过重定向将属性从一个请求传递到另一个请求

可以用一张图来整合上述关系

入门程序

编写两个简单的页面:

第一个:indel.jsp

<%--
  Created by IntelliJ IDEA.
  User: 17390
  Date: 2020/7/3
  Time: 11:16
  To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>


<h1>${1+4}</h1>
</body>
</html>

第二个info.jsp

这个页面用于反馈我们后端提供的信息。

<%--
  Created by IntelliJ IDEA.
  User: 17390
  Date: 2020/7/3
  Time: 14:10
  To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>

<h1> username:${username}</h1>

</body>
</html>

pom依赖

<?xml version="1.0" encoding="UTF-8"?>

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
  <modelVersion>4.0.0</modelVersion>

  <groupId>org.example</groupId>
  <artifactId>MVC01</artifactId>
  <version>1.0-SNAPSHOT</version>
  <packaging>war</packaging>

  <name>MVC01 Maven Webapp</name>
  <!-- FIXME change it to the project's website -->
  <url>http://www.example.com</url>

  <properties>
    <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
    <maven.compiler.source>1.7</maven.compiler.source>
    <maven.compiler.target>1.7</maven.compiler.target>
  </properties>

  <dependencies>
    <!--   SpringMVC -->
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-webmvc</artifactId>
      <version>5.2.2.RELEASE</version>
    </dependency>

    <!--JEE相关的-->
    <dependency>
      <groupId>javax.servlet</groupId>
      <artifactId>jstl</artifactId>
      <version>1.2</version>
    </dependency>
    <dependency>
      <groupId>javax.servlet.jsp</groupId>
      <artifactId>javax.servlet.jsp-api</artifactId>
      <version>2.3.3</version>
      <scope>provided</scope>
    </dependency>
    <dependency>
      <groupId>javax.servlet</groupId>
      <artifactId>javax.servlet-api</artifactId>
      <version>4.0.1</version>
      <scope>provided</scope>
    </dependency>
  </dependencies>
</project>

XML配置:

与在web项目中使用Spring中相同的是,我们也需要让SpringMVC随着web项目启动,SpringMVC的做法是利用DIspatcherServlet;

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
         version="4.0">

    <!--        注意要更改打包的方式-->
    <packaging>war</packaging>
    
    <servlet>
        <!--   配置mvc的核心控制器 -->
        <servlet-name>DispatcherServlet</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
		<!--  指定Spirng配置文件-->
        <init-param>
            <param-name>contextConfigLocation</param-name>
            <param-value>classpath:springmvc.xml</param-value>
        </init-param>
        <!-- 在应用程序启动时候,自动加载控制器-->
        <load-on-startup>0</load-on-startup>
    </servlet>

<!--    要交给SpringMVC处理的请求-->
    <servlet-mapping>
        <servlet-name>DispatcherServlet</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>
</web-app>

DispatcherServlet做了两个事情,1:初始化一个Spring容器,2:注册一个Servlet

SpringMVC本质上也是一个Spring容器,在不涉及WEB时使用方法和Spring没有任何区别;ß

Url-patten

只有请求地址能够匹配到到被DispatcherServlet的url-pattern的请求才会被SpringMVC处理,那么那些请求要交给SpringMVC处理呢,通常是除静态资源以外的请求;

常用pattern:

pattern 说明
/ 除了.jsp 以外的所有请求
/* 所有请求
*.action 所有 以action结尾的请求

需要说明的是action并不是固定的不同公司可能不同,但无论是点什么,其目的都是为了和静态资源加以区分

创建控制器类:

package controller;

import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.mvc.Controller;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

/**
 * Created by Jeason Luna on 2020/7/3 14:07
 */
public class UserController implements Controller {
    @Override
    public ModelAndView handleRequest(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse) throws Exception {

        httpServletRequest.setCharacterEncoding("UTF-8");
        System.out.println("aaaaaaaaaaaa!!!");
        ModelAndView modelAndView = new ModelAndView();
        //设置页面需要的数据
        modelAndView.addObject("username","jeason");
        //设置视图的文件名称
        modelAndView.setViewName("info.jsp");
		//返回模型和视图给dispatcherServlet
        return modelAndView;
    }
}

ModelAndView,其实就是把视图资源和数据打包到一起,然后视图名称交给视图解析器,Object放到请求中;

注册控制器到容器中:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">


    <!-- 通过映射名称(name)来告诉Spring这个Bean能处理哪个url
 		默认就是把请求映射到handle上面-->
    <bean name="/userInfo" class="controller.UserController"/>
</beans>

DispatcherServlet是按照请求路径来查找Handler,我们必须告诉SpringMVC,这个控制器是用来处理哪个请求地址的,默认情况下,SpringMVC会查找beanName与请求地址相同的Handler来处理;

测试:

打开网址:http://localhost:8080/day41_SpringMVCDemo1/ 可以得到我们的indel.jsp

打开网址:http://localhost:8080/day41_SpringMVCDemo1/userInfo 可以得到我们的indel.jsp

使用注解

虽然我们通过上述的方式实现了基本的功能,但是我们发现了一个问题,我们的SpringMVC的框架是帮助我们去整合各种请求操作的,我们现在并没有达成现在这样的需求,依然要一个url请求对应一个controller调用。即使我们让多个url请求去对应一个controller调用,因为controller类中只有一个方法,我们也不可能去实现不同调用对应不同的功能。

所以我们需要一种更为明晰的url和controller的对应关系,注解的方式可以帮助来实现这个需求。

我们重新建立一个控制类文件:

package controller;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.servlet.ModelAndView;

/**
 * Created by Jeason Luna on 2020/7/4 22:54
 */


@Controller
public class AnnotionController {

    @RequestMapping("/UserInfo")
    public ModelAndView UserInfo(){
        System.out.println("UserInfo is working!!!!!!");
        ModelAndView t = new ModelAndView();
        t.addObject("username","UserInfo is working!!!!!!");
        t.setViewName("info.jsp");
        return t;

    }

    @RequestMapping("/UserLogin")
    public ModelAndView UserLogin(){
        System.out.println("UserLogin is working!!!!!!");
        ModelAndView t = new ModelAndView();
        t.addObject("username","UserLogin is working!!!!!!");
        t.setViewName("info.jsp");
        return t;
    }
}

首先添加注解@Controller,在这个类中我们通过@RequestMapping的参数来指定url和调用方法间的映射关系,从而实现多个url映射不同的调用方法

注意,这里我们要注释原来放入Spring容器中的Bean对象,并且开启注解扫描,故配置文件修改如下:

<?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"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd">

<!--    <bean name="/useInfo" class="controller.UserController"/>-->

    <!-- 启动扫描注解-->
    <context:component-scan base-package="controller"/>
</beans>

测试:

打开网址:http://localhost:8080/day41_SpringMVCDemo1/UserInfo 可以得到我们的indel.jsp

打开网址:http://localhost:8080/day41_SpringMVCDemo1/UserLogin 可以得到

需要注意的是:
viewName路径若不带/时则从当前请求的位置查找文件,带/则表示从根路径查找

视图解析器配置

视图解析器用于查找视图文件,及生成视图对象,我们不用过多关注,唯一会用到的就是,为视图名称配置前缀和后缀从而简化,Handler中的书写

在一个实际项目中页面文件可能比较多,可以用文件夹管理,但是这导致我们在编写视图名称时更加繁琐,例如:

handler中:

这时就可在配置中对视图解析器进行相关设置;

<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
  <!--指定视图类型-->
    <property name="viewClass" value="org.springframework.web.servlet.view.JstlView"/>
  <!--指定前缀-->
    <property name="prefix" value="/pages/jsp/"/>
  <!--指定后缀-->
    <property name="suffix" value=".jsp"/>
</bean>

处理器中:

@RequestMapping("/getMsg")
public ModelAndView getMsg(){
    ModelAndView modelAndView = new ModelAndView();
    modelAndView.setViewName("index");
    modelAndView.addObject("msg","hello SpringMVC annotation!");
    return modelAndView;
}

视图解析器会自动在viewName前后分别拼接前缀和后缀;如:/pages/jsp/index.jsp

posted @ 2020-07-05 10:17  不愿透漏姓名的王建森  阅读(239)  评论(0编辑  收藏  举报