zoukankan      html  css  js  c++  java
  • java面试题之----spring MVC的原理和MVC

    1.什么是mvc?

    1.1原始比较初级的设计模式:

    1.2 MVC设计模式

    2MVC设计模式的优势核心在于其能解耦和:

    传统的设计模式相当于是一个串联的设计,只要其中一个环节出了问题便会使下一环节中止

    网上普遍的对于mvc设计模式的解读为:

    2.1.定义:MVC 设计模型是一种使用 Model View Controller( 模型-视图-控制器)设计创建 Web 应用程序的模式。

      由上主谓宾可以很容易看出,mvc模型是一种用来写web应用程序的样式,也就是说只能写web不能写其它?

    2.2 既然使用了 Model View Controller( 模型-视图-控制器),那么就很有必要来介绍一下该(模型-视图-控制器)到底是怎样的一个东西?

    Model(模型):是应用程序中用于处理应用程序数据逻辑的部分。

        通常模型对象负责在数据库中存取数据。
    View(视图):是应用程序中处理数据显示的部分。
        通常视图是依据模型数据创建的。
    Controller(控制器):是应用程序中处理用户交互的部分。
        通常控制器负责从视图读取数据,控制用户输入,并向模型发送数据。
    MVC的流程:
    用户向服务器发送请求,控制层(Controller)接受请求,并委托模型层(Model)进行数据处理,模型层接到委托后处理请求并将结果返回给控制层(Controller),并将返回的结果交给视图层(view),让视图层(view)进行视图渲染,并将渲染后的jstl返回给控制层(Controller),最后控制层(Controller)响应结果请求返回给用户。便可以显示浏览器页面了,其中C/S模式,模型层会主动推送数据到视图层,而B/S则不能
    最后终于是我们的SpringMVC了,
    到底什么是Spring MVC呢?
    Spring MVC是一个web层的mvc框架类似于struts2

    3 Springmvc和spring?

    Springmvc是spring的部分。

     

    4.Springmvc执行流程

    根据mvc设计模式:

    用户向服务器发送请求,前端控制器(web.xml)接受请求,前端控制器将请求分为如下几步

    1.向处理器映射器(HandlerMapping)请求获取action类的url,处理器映射器(springmvc.xml)获取到action类的url后便将带有拦截器的chian返回给前端控制器

    2.向处理器适配器(handlerAdapter)请求执行action,处理器适配器(springmvc.xml)执行action的方法,所以action需要调用业务层的方法,并将返回结果(mode and view)返回给处理器适配器,处理器适配器将得到的结果(mode and view)返回给前端控制器。

    3.向视图解析器(ViewResolver)请求视图解析,视图解析器(springmvc.xml)接受到请求后立即解析视图,并将view返回给前端控制器。

    4.前端控制器接收到view后向视图层(view)发送视图渲染请求,视图层渲染完后将jstl返回给前端控制器,

    5.最后前端控制器将得到的结果响应返回给用户。

     
    springmvc中mvc各自的职责
    DispatcherServlet前端控制器设计模式的实现,提供Spring Web MVC的集中访问点,而且负责职责的分派,而且与Spring IoC容器无缝集成,从而可以获得Spring的所有好处。

    DispatcherServlet主要用作职责调度工作,本身主要用于控制流程,主要职责如下:

    1、文件上传解析,如果请求类型是multipart将通过MultipartResolver进行文件上传解析;

    2、通过HandlerMapping,将请求映射到处理器(返回一个HandlerExecutionChain,它包括一个处理器、多个HandlerInterceptor拦截器);

    3、  通过HandlerAdapter支持多种类型的处理器(HandlerExecutionChain中的处理器);

    4、通过ViewResolver解析逻辑视图名到具体视图实现;

    5、本地化解析;

    6、渲染具体的视图等;

    7、如果执行过程中遇到异常将交给HandlerExceptionResolver来解析。
    视图层

    视图是用户看到并与之交互的界面。对老式的Web应用程序来说,视图就是由HTML元素组成的界面,在新式的Web应用程序中,HTML依旧在视图中扮演着重要的角色,但一些新的技术已层出不穷,它们包括Macromedia Flash和像XHTML,XML/XSL,WML等一些标识语言和Web services.   如何处理应用程序的界面变得越来越有挑战性。MVC一个大的好处是它能为你的应用程序处理很多不同的视图。在视图中其实没有真正的处理发生,不管这些数据是联机存储的还是一个雇员列表,作为视图来讲,它只是作为一种输出数据并允许用户操纵的方式。

    模型层:

    模型表示企业数据和业务规则。在MVC的三个部件中,模型拥有最多的处理任务。例如它可能用像EJBs和ColdFusion Components这样的构件对象来处理数据库。被模型返回的数据是中立的,就是说模型与数据格式无关,这样一个模型能为多个视图提供数据。由于应用于模型的代码只需写一次就可以被多个视图重用,所以减少了代码的重复性。

    mvc设计模式的优点

    1、开发人员可以只关注整个结构中的其中某一层;

    2、可以很容易的用新的实现来替换原有层次的实现;

    3、可以降低层与层之间的依赖;

    4、有利于标准化;

    5、利于各层逻辑的复用。

    spring具体流程回顾及相应代码

    具体流程:

    (1)首先用户发送请求——>DispatcherServlet,前端控制器收到请求后自己不进行处理,而是委托给其他的解析器进行处理,作为统一访问点,进行全局的流程控制;

    (2)DispatcherServlet——>HandlerMapping,映射处理器将会把请求映射为HandlerExecutionChain对象(包含一个Handler处理器(页面控制器)对象、多个HandlerInterceptor拦截器)对象;

    (3)DispatcherServlet——>HandlerAdapter,处理器适配器将会把处理器包装为适配器,从而支持多种类型的处理器,即适配器设计模式的应用,从而很容易支持很多类型的处理器;

    (4)HandlerAdapter——>调用处理器相应功能处理方法,并返回一个ModelAndView对象(包含模型数据、逻辑视图名);

    (5)ModelAndView对象(Model部分是业务对象返回的模型数据,View部分为逻辑视图名)——> ViewResolver, 视图解析器将把逻辑视图名解析为具体的View;

    (6)View——>渲染,View会根据传进来的Model模型数据进行渲染,此处的Model实际是一个Map数据结构;

    (7)返回控制权给DispatcherServlet,由DispatcherServlet返回响应给用户,到此一个流程结束。

    SpringMVC入门程序

    (1)web.xml

    <web-app>
      <servlet>
      <!-- 加载前端控制器 -->
      <servlet-name>springmvc</servlet-name>
      <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
      <!-- 
           加载配置文件
           默认加载规范:
           * 文件命名:servlet-name-servlet.xml====springmvc-servlet.xml
           * 路径规范:必须在WEB-INF目录下面
           修改加载路径:
       -->
       <init-param>
       <param-name>contextConfigLocation</param-name>
       <param-value>classpath:springmvc.xml</param-value>   
       </init-param>
      </servlet>
    
      <servlet-mapping>
      <servlet-name>springmvc</servlet-name>
      <url-pattern>*.do</url-pattern>
      </servlet-mapping>
    </web-app>
    

    (2)springmvc.xml

    <beans>
        <!-- 配置映射处理器:根据bean(自定义Controler)的name属性的url去寻找hanler;springmvc默认的映射处理器是
        BeanNameUrlHandlerMapping
         -->
        <bean class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping"></bean>
    
    
        <!-- 配置处理器适配器来执行Controlelr ,springmvc默认的是
        SimpleControllerHandlerAdapter
        -->
        <bean class="org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter"></bean>
    
        <!-- 配置自定义Controler -->
        <bean id="myController" name="/hello.do" class="org.controller.MyController"></bean>
    
        <!-- 配置sprigmvc视图解析器:解析逻辑试图; 
            后台返回逻辑试图:index
            视图解析器解析出真正物理视图:前缀+逻辑试图+后缀====/WEB-INF/jsps/index.jsp
        -->
        <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
            <property name="prefix" value="/WEB-INF/jsps/"></property>
            <property name="suffix" value=".jsp"></property>        
        </bean>
    </beans>
    

    (3)自定义Controler

    public class MyController implements Controller{
    
        public ModelAndView handleRequest(HttpServletRequest arg0,
                HttpServletResponse arg1) throws Exception {
            ModelAndView mv = new ModelAndView();
            //设置页面回显数据
            mv.addObject("hello", "欢迎学习springmvc!");
    
            //返回物理视图
            //mv.setViewName("/WEB-INF/jsps/index.jsp");
    
            //返回逻辑视图
            mv.setViewName("index");
            return mv;
        }
    }
    

    (4)index页面

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

    (5)测试地址

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

    HandlerMapping

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

    映射处理器有三种,三种可以共存,相互不影响,分别是BeanNameUrlHandlerMapping、SimpleUrlHandlerMapping和ControllerClassNameHandlerMapping;

    BeanNameUrlHandlerMapping

    //默认映射器,即使不配置,默认就使用这个来映射请求。
    <bean class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping"></bean>
    //映射器把请求映射到controller
    <bean id="testController" name="/hello.do" class="org.controller.TestController"></bean>
    

    SimpleUrlHandlerMapping

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

    ControllerClassNameHandlerMapping

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

    HandlerMapping架构图

    HandlerAdapter

    处理器适配器有两种,可以共存,分别是SimpleControllerHandlerAdapter和HttpRequestHandlerAdapter。

    SimpleControllerHandlerAdapter
    SimpleControllerHandlerAdapter是默认的适配器,表示所有实现了org.springframework.web.servlet.mvc.Controller 接口的Bean 可以作为SpringMVC 中的处理器。

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

    (1)配置HttpRequestHandlerAdapter适配器

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

    (2)编写Controller

    public class HttpController implements HttpRequestHandler{
    
        public void handleRequest(HttpServletRequest request, HttpServletResponse response)
                throws ServletException, IOException {
            //给Request设置值,在页面进行回显
            request.setAttribute("hello", "这是HttpRequestHandler!");
            //跳转页面
            request.getRequestDispatcher("/WEB-INF/jsps/index.jsp").forward(request, response);
        }
    }
    

    (3)准备jsp页面

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

    Adapter源码分析
    模拟场景:前端控制器(DispatcherServlet)接收到Handler对象后,传递给对应的处理器适配器(HandlerAdapter),处理器适配器调用相应的Handler方法。

    (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..");
        }
    } 
    

    (2)模拟HandlerAdapter

    //以下是HandlerAdapter接口和它的三种实现
    public interface HandlerAdapter {
        public boolean supports(Object handler);
        public void handle(Object handler);
    }
    
    public class SimpleHandlerAdapter implements HandlerAdapter{
        public boolean supports(Object handler) {
            return (handler instanceof SimpleController);
        }
    
        public void handle(Object handler) {
            ((SimpleController)handler).doSimpleHandler();
        }
    }
    
    public class HttpHandlerAdapter implements HandlerAdapter{
        public boolean supports(Object handler) {
            return (handler instanceof HttpController);
        }
    
        public void handle(Object handler) {
            ((HttpController)handler).doHttpHandler();
        }
    }
    
    public class AnnotationHandlerAdapter implements HandlerAdapter{
        public boolean supports(Object handler) {
            return (handler instanceof AnnotationController);
        }
    
        public void handle(Object handler) {
            ((AnnotationController)handler).doAnnotationHandler();
        }
    }
    

    (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;
        }
    }
    

    (4)测试

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

    控制器

    控制器架构图

    Controller 简介

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

    ServletForwardingController(转发控制器)

    将接收到的请求转发到一个命名的servlet,具体示例如下:当我们请求/forwardToServlet.do时,会被转发到名字为“forwarding”的servlet处理,该sevlet的servlet-mapping标签配置是可选的.

    (1)控制器

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

    (2)web.xml

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

    (3)spring.xml

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

    AbstractCommandController(命令控制器)

    使用post请求进行表单提交

    模拟提交用户表信息。

    (1)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>
    

    (2)表单跳转控制器:跳转到表单页面

    public class ToAddController implements Controller{
        public ModelAndView handleRequest(HttpServletRequest request,
                HttpServletResponse response) throws Exception {
            ModelAndView mv = new ModelAndView();
            //调转到add添加页面视图
            mv.setViewName("add");
            return mv;
        }
    }
    

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

    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));    
            }
        }
    }
    

    (4)表单页面:

    <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>
    

    (5)表单信息呈现页面:

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

    (6)进入表单页面

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

    使用get请求进行表单提交

    在上面的代码基础上,直接输入地址:

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

    ParameterizableViewController(参数控制器)

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

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

    中文乱码解决

    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>

     总结:

    在面试中问到这个问题后,如果将mvc和springmvc的执行流程和整体结构讲述清楚便可以了

  • 相关阅读:
    LeetCode 93. Restore IP Addresses
    LeetCode 92. Reverse Linked List II
    LeetCode 94. Binary Tree Inorder Traversal
    javaweb中重定向和请求转发(response.sendRedirect()和request.getRequestDispatcher(rul).forward(request,response)))的区别
    java关于jdbc的配置与使用步骤
    关于php中的include html文件的问题,为什么html可以在php中执行
    yii2 无法显示debug条的问题解决方法
    elasticsearch报错expected <block end>, but found BlockMappingStart解决方法
    sysctl -p 报错问题的解决方法
    yii2 Rbac使用yii命令一键建表
  • 原文地址:https://www.cnblogs.com/xuxinstyle/p/9391791.html
Copyright © 2011-2022 走看看