java面试题之 |
您所在的位置:网站首页 › mvc模式和springmvc区别 › java面试题之 |
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 springmvc org.springframework.web.servlet.DispatcherServlet contextConfigLocation classpath:springmvc.xml springmvc *.do(2)springmvc.xml (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页面 ${hello}(5)测试地址 http://localhost:8080/springmvc/hello.do HandlerMappingHandlerMapping 将会把请求映射为 HandlerExecutionChain 对象(包含一个 Handler 处理器(页面控制器)对象、多个 HandlerInterceptor 拦截器)对象,通过这种策略模式,很容易添加新的映射策略。 映射处理器有三种,三种可以共存,相互不影响,分别是BeanNameUrlHandlerMapping、SimpleUrlHandlerMapping和ControllerClassNameHandlerMapping; BeanNameUrlHandlerMapping //默认映射器,即使不配置,默认就使用这个来映射请求。 //映射器把请求映射到controllerSimpleUrlHandlerMapping testController testController //那么上面的这个映射配置:表示多个*.do文件可以访问多个Controller或者一个Controller。 //前提是:都必须依赖自定义的控制器beanControllerClassNameHandlerMapping //这个Mapping一配置:我们就可以使用Contrller的 [类名.do]来访问这个Controller.HandlerMapping架构图
处理器适配器有两种,可以共存,分别是SimpleControllerHandlerAdapter和HttpRequestHandlerAdapter。 SimpleControllerHandlerAdapter SimpleControllerHandlerAdapter是默认的适配器,表示所有实现了org.springframework.web.servlet.mvc.Controller 接口的Bean 可以作为SpringMVC 中的处理器。 HttpRequestHandlerAdapter HTTP请求处理器适配器将http请求封装成HttpServletResquest 和HttpServletResponse对象,和servlet接口类似。 (1)配置HttpRequestHandlerAdapter适配器 (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页面 ${hello}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 = new ArrayList(); 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 forwarding org.controller.ForwardingServlet(3)spring.xml AbstractCommandController(命令控制器) 使用post请求进行表单提交 模拟提交用户表信息。 (1)spring.xml配置文件: (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)表单页面: 姓名:生日: 性别: 地址:
(5)表单信息呈现页面: ${user.username } ${user.birthday } ${user.sex } ${user.address }(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进行访问。 中文乱码解决Get请求乱码 对于get请求中文参数出现乱码解决方法有两个: 修改tomcat配置文件添加编码与工程编码一致,如下: 另外一种方法对参数进行重新编码: String userName =new String(request.getParamter("userName").getBytes("ISO8859-1"),"UTF-8") ISO8859-1是Tomcat默认编码,需要将Tomcat编码后的内容按UTF-8编码Post请求乱码 在web.xml中加入: characterEncoding org.springframework.web.filter.CharacterEncodingFilter encoding UTF-8 characterEncoding /*
总结: 在面试中问到这个问题后,如果将mvc和springmvc的执行流程和整体结构讲述清楚便可以了 |
今日新闻 |
推荐新闻 |
CopyRight 2018-2019 办公设备维修网 版权所有 豫ICP备15022753号-3 |