spring mvc

一、介绍

        JavaEE体系结构包括四层,从上到下分别是应用层、Web层、业务层、持久层。Struts/Struts2和SpringMVC是

Web层的框架,Spring是业务层的框架,Hibernate和MyBatis是持久层的框架。其中springmvc是Spring中的一部分。

1.作用

   SpringMVC是一种基于Java,实现了Web MVC设计模式,请求驱动类型的轻量级Web框架,即使用了MVC架构模式的思想,将Web层进行职责解耦。基于请求驱动指的就是使用请求-响应模型,框架的目的就是帮助我们简化开发,SpringMVC也是要简化我们日常Web开发。

2.MVC设计模式

              

 

              

3.spring mvc的核心架构

               

   具体流程:

        ①用户发送请求---> DispatcherServlet:

           前端控制器接收到请求之后自己不进行处理,而是将请求委托给其他解析器进行处理,作为统一访问点,进行全

           局的流程控制。            

        ②DispatcherServlet---> HandlerMapping:

            映射处理器将会把请求映射为HandlerExecutionChain对象(包括一个Handler处理器(页面控制器)对象、多个
            HandlerInterceptor拦截器)

        ③DispatcherServlet---> HandlerAdapter:

            处理器适配器将会把处理器包装成适配器,从而支持多种类型的处理器,即适配器模式的应用,从而很容器支持

            多种处理器。

        ④HandlerAdapter---> 调用处理器相应功能处理方法:

           处理完数据之后,并返回一个ModelAndView对象(包括模型数据,逻辑视图名)。

        ⑤ModelAndView对象(Model部分是模型数据,View部分是逻辑视图名)--->ViewResolver:

           视图解析器将逻辑视图名解析为具体的View。

        ⑥view---> 渲染:

           view会根据传进来的Model模型数据进行渲染,此时的Model实际是一个Map数据结构。

        ⑦DispatcherServlet----> 用户:

            DspatcherServlet将视图返回响应给用户。

二、spring mvc核心

1.HandlerMapping(映射处理器)

HandlerMapping 将会把请求映射为 HandlerExecutionChain 对象(包含一个 Handler 处理器(页面控制器)对象、多个 HandlerInterceptor 拦截器)对象,通过这种策略模式,很容易添加新的映射策略。

映射处理器有三种,三种可以共存,相互不影响:

BeanNameUrlHandlerMapping:

<!--默认映射器,即使不配置,默认就使用这个来映射请求。-->
<bean class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping"></bean>

<!--不使用注解定义COntroller bean时 -->
<bean id="springmvccontroller" name="/name" class="com.spring.SpringmvcController"></bean>

SimpleUrlHandlerMapping:

//那这个映射配置:表示多个*.do文件可以访问多个Controller或者一个Controller。 
//前提是:都必须依赖自定义的控制器bean
<bean class="org.springframework.web.servlet.handler.SimpleUrlHandlerMapping"> 
     <property name="mappings">
            <props> 
                <prop key="/ss.do">springmvcController</prop> 
                <prop key="/abc.do">springmvcController</prop> 
            </props> </property> 
</bean>

<!--不使用注解定义COntroller bean时 -->
<bean id="springmvccontroller" name="/name" class="com.spring.SpringmvcController"></bean>

ControllerClassNameHandlerMapping:

<!--这个Mapping一配置:我们就可以使用Contrller的 [类名.do]来访问这个Controller -->
 <bean   class="org.springframework.web.servlet.mvc.support.ControllerClassNameHandlerMapping"></bean>

2.HandlerAdapter

处理器适配器有两种:
SimpleControllerHandlerAdapter:
          SimpleControllerHandlerAdapter是默认的适配器,表示所有实现了org.springframework.web.servlet.mvc.Controller

          接口的Bean 可以作为SpringMVC 中的处理器。

HttpRequestHandlerAdapter:
HTTP请求处理器适配器将http请求封装成HttpServletResquest 和HttpServletResponse对象,和servlet接口类似。

  配置:

<!-- 配置HttpRequestHandlerAdapter适配器 -->
<bean class="org.springframework.web.servlet.mvc.HttpRequestHandlerAdapter"></bean>

  对应的Controller:

public class HttpController implements HttpRequestHandler{ 

     public void handleRequest(HttpServletRequest request, HttpServletResponse response)   
          throws ServletException, IOException { 

         //给Request设置值,在页面进行回显 
         request.setAttribute("name", "这是HttpRequestHandler!"); 
         //跳转页面 
         request.getRequestDispatcher("/WEB-INF/jsps/name.jsp").forward(request, response);
    } 
}

  jsp:
 

<html>
<body>
<h1>${name}</h1>
</body>
</html>

3.常用注解

@Controller

  负责注册一个bean 到spring 上下文中
@RequestMapping

  注解为控制器指定可以处理哪些 URL 请求。
@RequestBody

  该注解用于读取Request请求的body部分数据,使用系统默认配置的HttpMessageConverter进行解析,然后把相

       应的数据绑定到要返回的对象上 ,再把HttpMessageConverter返回的对象数据绑定到 controller中方法的参数上

@ResponseBody。

   该注解用于将Controller的方法返回的对象,通过适当的HttpMessageConverter转换为指定格式后,写入到

        Response对象的body数据区。

@ModelAttribute    

  在方法定义上使用 @ModelAttribute 注解:Spring MVC 在调用目标处理方法前,会先逐个调用在方法级上标注了@ModelAttribute 的方法

  在方法的入参前使用 @ModelAttribute 注解:可以从隐含对象中获取隐含的模型数据中获取对象,再将请求参数 –

       绑定到对象中,再传入入参将方法入参对象添加到模型中。

@RequestParam 

  在处理方法入参处使用 @RequestParam 可以把请求参 数传递给请求方法。

@PathVariable

  绑定 URL 占位符到入参。
@ExceptionHandler

  注解到方法上,出现异常时会执行该方法。
@ControllerAdvice

  使一个Contoller成为全局的异常处理类,类中用@ExceptionHandler方法注解的方法可以处理所有Controller发生

       的异常。

三、实例

①简单示例:

          1.创建 web project项目

          2.导包

            

          3.配置

            3.1 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/javaee" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd" id="WebApp_ID" version="3.0">
  <display-name>springmvc</display-name>
  <welcome-file-list>
    <welcome-file>index.html</welcome-file>
    <welcome-file>index.htm</welcome-file>
    <welcome-file>index.jsp</welcome-file>
    <welcome-file>default.html</welcome-file>
    <welcome-file>default.htm</welcome-file>
    <welcome-file>default.jsp</welcome-file>
  </welcome-file-list>
  
  <!--加载前端控制器-->
  <!--configure the setting of springmvcDispatcherServlet and configure the mapping-->
  <servlet>
      <servlet-name>springmvc</servlet-name>
      <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>

      <!-- 加载配置文件 默认加载规范:
                  * 文件命名:servlet-name-servlet.xml====springmvc-servlet.xml 
                  * 路径规范:必须在WEB-INF目录下面 
           修改加载路径:以下是在src根目录非默认下,所以需要配置该节点 -->
      <init-param>
            <param-name>contextConfigLocation</param-name>
            <param-value>classpath:springmvc-servlet.xml</param-value>
        </init-param>
        <!-- <load-on-startup>1</load-on-startup> -->
  </servlet>

  <servlet-mapping>
      <servlet-name>springmvc</servlet-name>
      <url-pattern>/</url-pattern>
  </servlet-mapping>
</web-app>

          3.2 springmvc-servlet.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 http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.1.xsd
        http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-4.1.xsd">                    

    <!-- scan the package and the sub package -->
    <context:component-scan base-package="com.springmvc"/>

    <!-- don't handle the static resource -->
    <!-- 它会像一个检查员,对进入DispatcherServlet的URL进行筛查,如果发现是静态资源的请求,就将该请
         求转由Web应用服务器默认的Servlet处理,如果不是静态资源的请求,才由DispatcherServlet继续处理 -->
    <mvc:default-servlet-handler />
    
    <!-- if you use annotation you must configure following setting -->
    <!-- 该配置实际做的是采用默认自动加载映射处理器和处理器适配器,使用该方式时不需要配置下面的1和2-->
    <mvc:annotation-driven />
    
    <!-- 1:配置映射处理器:根据bean(自定义Controler)的name属性的url去寻找hanler;
    springmvc默认的映射处理器是 BeanNameUrlHandlerMapping,不配置的话会自动加载默认的 
    <bean class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping"></bean> -->
  
    <!--:2:配置处理器适配器来执行Controlelr ,springmvc默认的是 SimpleControllerHandlerAdapter  
    <bean class="org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter"></bean>-->
    
     <!--3: 配置sprigmvc视图解析器:解析逻辑试图; 
        后台返回逻辑试图:index
        视图解析器解析出真正物理视图:前缀+逻辑试图+后缀====/WEB-INF/jsps/index.jsp -->
    <!-- configure the InternalResourceViewResolver -->
    <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" 
            id="internalResourceViewResolver">
        <!-- 前缀 -->
        <property name="prefix" value="/WEB-INF/springmvc/jsps/" />
        <!-- 后缀 -->
        <property name="suffix" value=".jsp" />
    </bean>
</beans>

         3.3 在WEB-INF文件夹下创建springmvc/jsps文件夹(与prefix对应的保持一致即可),用来存放jsp视图。并创建name.jsp

<%@ page language="java" pageEncoding="UTF-8"%>
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
<head></head>

<body>
<a style="color:red">
    我叫破天,来自湖北黄石市阳新县黄双口镇金星村六组</a>
</body>
</html>

  注意:在jsp组件中使用<%@ page language="java" pageEncoding="UTF-8"%>时需要导入 tomcat liberary,

             具体见  https://blog.youkuaiyun.com/weixin_41690908/article/details/85058417    

         3.4 创建SpringmvcController

package com.springmvc;

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

/*
*
*@Controller :负责注册一个bean 到spring 上下文中
*
* @RequestMapping :注解为控制器指定可以处理哪些 URL 请求
*/

@Controller              
@RequestMapping("/springmvc")   
public class SpringmvcController {
	
	   @RequestMapping("/name")
	    public String name(){     
		   System.out.println("石破天");
	        return "name";
	    }

}

        3.5 测试

              启动服务器之后在浏览器地址栏输入 http://localhost:8080/springmvc/springmvc/name  进行访问

            

四、其他知识点解析

 1.Adapter源码分析

    模拟场景:前端控制器(DispatcherServlet)接收到Handler对象后,传递给对应的处理器适配器

 (HandlerAdapter),处理器适配器调用相应的Handler方法。

  1.1 模拟Controller

//以下是Controller接口和它的三种实现 
public interface Controller { 
...
} 

public class SimpleController implements Controller{ 
    public void doSimpleHandler() {
        System.out.println("Simple..."); 
    } 
} 

public class HttpController implements Controller{ 

    public void doHttpHandler() { 
        System.out.println("Http...");
    } 
}

public class AnnotationController implements Controller{
    public void doAnnotationHandler() { 
        System.out.println("Annotation.."); 
    } 
} 

1.2 模拟HandlerAdapter

//以下是HandlerAdapter接口和它的三种实现
public interface HandlerAdapter { 
    public boolean supports(Object handler); 
    public void handle(Object handler); 
} 

//SimpleHandlerAdapter处理器适配器
public class SimpleHandlerAdapter implements HandlerAdapter{ 
    public boolean supports(Object handler) { 
        return (handler instanceof SimpleController); 
    } 
    public void handle(Object handler) { 
        ((SimpleController)handler).doSimpleHandler(); 
    } 
}

//HttpHandlerAdapter处理器适配器
public class HttpHandlerAdapter implements HandlerAdapter{ 
    public boolean supports(Object handler) { 
        return (handler instanceof HttpController);
    } 
    public void handle(Object handler) {
        ((HttpController)handler).doHttpHandler(); 
    } 
} 

//AnnotationHandlerAdapter处理器适配器
public class AnnotationHandlerAdapter implements HandlerAdapter{ 
    public boolean supports(Object handler) { 
        return (handler instanceof AnnotationController);
    } 
    public void handle(Object handler) { 
       ((AnnotationController)handler).doAnnotationHandler();
    } 
}

1.3 模拟DispatcherServlet

public class Dispatcher { 
    public static List<HandlerAdapter> handlerAdapter = new ArrayList<HandlerAdapter>(); 
    public Dispatcher(){ 
        handlerAdapter.add(new SimpleHandlerAdapter()); 
        handlerAdapter.add(new HttpHandlerAdapter()); 
        handlerAdapter.add(new AnnotationHandlerAdapter()); 
    }

    //核心功能 
    public void doDispatch() { 

    //前端控制器(DispatcherServlet)接收到Handler对象后 
    //SimpleController handler = new SimpleController(); 
    //HttpController handler = new HttpController(); 
    AnnotationController handler = new AnnotationController(); 
    
    //传递给对应的处理器适配器(HandlerAdapter) 
    HandlerAdapter handlerAdapter = getHandlerAdapter(handler); 
    
    //处理器适配器调用相应的Handler方法 
    handlerAdapter.handle(handler); } 

    //通过Handler找到对应的处理器适配器(HandlerAdapter)
    public HandlerAdapter getHandlerAdapter(Controller handler) { 
        for(HandlerAdapter adapter : handlerAdapter){ 
            if(adapter.supports(handler)){ 
                return adapter; 
              } 
            } 
        return null; 
   }
} 

1.4 测试

public class Test { 
    public static void main(String[] args) { 
        Dispatcher dispather = new Dispatcher(); 
        dispather.doDispatch();
    } 
}

 2 Controller

    作用:

  • 收集、验证请求参数并绑定到命令对象;
  • 将命令对象交给业务对象,由业务对象处理并返回模型数据;
  • 返回ModelAndView(Model部分是业务对象返回的模型数据,视图部分为逻辑视图名)。

2.1 ServletForwardingController(转发控制器)

 将接收到的请求转发到一个命名的servlet,具体示例如下:当我们请求/forwardToServlet.do时,会被转发到

名字为“forwarding”的servlet处理,该sevlet的servlet-mapping标签配置是可选的。

配置示例:

    控制器:

public class ForwardingServlet extends HttpServlet{ 
    @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) 
        throws ServletException, IOException { 
        resp.getWriter().write("Controller forward to Servlet"); 
    }
}

 web.xml

servlet>  
<servlet-name>forwarding</servlet-name>  
<servlet-class>org.controller.ForwardingServlet</servlet-class>  
</servlet> 

  springmvc-servlet.xml

<bean name="/forwardToServlet.do"
    class="org.springframework.web.servlet.mvc.ServletForwardingController"> 
      <property name="servletName" value="forwarding"></property> 
</bean>

2.2 AbstractCommandController(命令控制器)

分两种情况:

    使用post请求进行表单提交

    模拟提交用户表信息。

    spring.xml配置文件:

<beans> 
   <!-- 配置映射处理器-->
   <bean class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping"></bean>    
   
   <!-- 配置处理器适配器--> 
   <bean  class="org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter</bean>    
   
   <!-- 配置自定义Controler --> 
   <bean name="/command.do" class="org.controller.CommandController"></bean>
   <bean name="/toAdd.do" class="org.controller.ToAddController"></bean> 
   
   <!-- 配置sprigmvc视图解析器:解析逻辑试图 --> 
   <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">   
           <property name="prefix" value="/WEB-INF/jsps/"></property> 
           <property name="suffix" value=".jsp"></property> 
   </bean> 

</beans>

    表单跳转控制器:跳转到表单页面

public class ToAddController implements Controller{ 
    public ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse 
        response) throws Exception { 
        ModelAndView mv = new ModelAndView(); 

        //调转到add添加页面视图 
        mv.setViewName("add"); 
        return mv; 
     }
}

  编辑页面控制器:转发表单信息

public class CommandController extends AbstractCommandController{ 

//指定参数绑定到哪个
    javaBean public CommandController(){ 
    this.setCommandClass(User.class); } 
    @Override protected ModelAndView handle(HttpServletRequest request, 
        HttpServletResponse response, Object command, BindException errors) throws Exception { 
        //把命令对象强转成User对象 User user = (User) command; 
        ModelAndView mv = new ModelAndView(); 
        mv.addObject("user", user); mv.setViewName("MyJsp"); 
        return mv; 
    } 

    /* * 进行时间类型各种格式的覆盖 */ 
    @Override protected void initBinder(HttpServletRequest request,  
        ServletRequestDataBinder binder) throws Exception { 
        
        String str = request.getParameter("birthday");
        if(str.contains("/")){ 
            binder.registerCustomEditor(Date.class, new CustomDateEditor(new      SimpleDateFormat("yyyy/MM/dd"), true));
        }else{ 
            binder.registerCustomEditor(Date.class, new CustomDateEditor(new  SimpleDateFormat("yyyy-MM-dd"), true));
        } 
    }
}

 表单页面

<html> 
  <body> 
    <form action="${pageContext.request.contextPath }/command.do" method="post"> 
      姓名:<input type="text" name="username" id="username"><p>
      生日:<input type="text" name="birthday" id="birthday"><p> 
      性别:<input type="text" name="sex" id="sex"><p> 
      地址:<input type="text" name="address" id="address"><p> 
           
      <input type="submit" value="提交"> 
    </form> 
  </body> 
</html>

 表单信息呈现页面

<html>
<body>
${user.username } <br>
${user.birthday } <br>
${user.sex } <br>
${user.address } <br>
</body>
</html>

启动服务器之后地址栏中

http://localhost:8080/springmvc/toAdd.do 

   使用get请求进行表单提交

http://localhost:8080/springmvc/command.do?username=ltx&birthday=1996/11/01&sex=男&address=广东

2.3 ParameterizableViewController(参数控制器)

     使用参数控制器,不用自己定义Controller,可以直接使用toIndex进行访问。

<bean name="/toIndex.do" class="org.springframework.web.servlet.mvc.ParameterizableViewController">
     <!-- 配置你所要跳转到视图的名称。跳转到index页面--> 
    <property name="viewName" value="index"></property>
</bean>

3 中文乱码

   Get请求乱码:

    对于get请求中文参数出现乱码解决方法有两个:

   ①修改tomcat配置文件添加编码与工程编码一致,如下:

          <Connector URIEncoding="UTF-8" connectionTimeout="20000" port="8080" protocol="HTTP/1.1" redirectPort="8443"/> 
   ②另外一种方法对参数进行重新编码:

         String userName =new String(request.getParamter("userName").getBytes("ISO8859-1"),"UTF-8")

         ISO8859-1是Tomcat默认编码,需要将Tomcat编码后的内容按UTF-8编码。


Post请求乱码

   在web.xml中加入:(配置过滤器)

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

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值