SpringMVC基础——返回值问题

一、SpringMVC 使用 ModelAndView 来处理返回值问题。

1.ModelAndView

官方描述:


Holder for both Model and View in the web MVC framework.
Note that these are entirely distinct. This class merely holds
both to make it possible for a controller to return both model
and view in a single return value.

<p>Represents a model and view returned by a handler, to be resolved
by a DispatcherServlet. The view can take the form of a String
view name which will need to be resolved by a ViewResolver object;
alternatively a View object can be specified directly. The model
is a Map, allowing the use of multiple objects keyed by name.

说明一下:

在 springmvc 框架中,ModelAndView 表示同时持有 Model 和 View 。Model 和 View 是完全不同的。

这个类使一个控制器返回单独的一个值同时包含 model 和 view 成为一种可能。

同时也代表着一个处理器返回一个 model 和 view ,会被 DispatcherServlet 解析。

View 对象如果是通过一个字符串形式的 view name 获取到的,则能被 ViewResolver 对象解析。

或者也可以直接指定 View 对象。model 是一个 Map 类型,可以使用多个对象的键值对。

2.这里要搞明白一点,为什么说是通过 ModelAndView 来处理返回值问题,事实上,返回的类型可以是很多种,可以是 ModelAndView 类型,

也可以是String类型,还可以是View类型,还有很多。但是他们最终会被解析为 ModelAndView 类型的对象。

通过源码来证实一下:

org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter#invokeHandlerMethod

在这个方法中:

Object result = methodInvoker.invokeHandlerMethod(handlerMethod, handler, webRequest, implicitModel);
ModelAndView mav = methodInvoker.getModelAndView(handlerMethod, handler.getClass(), result, implicitModel, webRequest);

其中 result 就是我们调用 handler 方法后的返回值。

通过 mehtodInvoker.getModelAndView() 方法将 result 最终解析为 ModelAndView 对象。

来看看具体过程:

public ModelAndView getModelAndView(Method handlerMethod, Class<?> handlerType, Object returnValue,
                ExtendedModelMap implicitModel, ServletWebRequest webRequest) throws Exception {
    ResponseStatus responseStatusAnn = AnnotationUtils.findAnnotation(handlerMethod, ResponseStatus.class);
    if (responseStatusAnn != null) {
        HttpStatus responseStatus = responseStatusAnn.value();
        String reason = responseStatusAnn.reason();
        if (!StringUtils.hasText(reason)) {
            webRequest.getResponse().setStatus(responseStatus.value());
        }
        else {
            webRequest.getResponse().sendError(responseStatus.value(), reason);
        }

        // to be picked up by the RedirectView
        webRequest.getRequest().setAttribute(View.RESPONSE_STATUS_ATTRIBUTE, responseStatus);

        responseArgumentUsed = true;
    }

    // Invoke custom resolvers if present...
    if (customModelAndViewResolvers != null) {
        for (ModelAndViewResolver mavResolver : customModelAndViewResolvers) {
            ModelAndView mav = mavResolver.resolveModelAndView(
                    handlerMethod, handlerType, returnValue, implicitModel, webRequest);
            if (mav != ModelAndViewResolver.UNRESOLVED) {
                return mav;
            }
        }
    }

    if (returnValue instanceof HttpEntity) {
        handleHttpEntityResponse((HttpEntity<?>) returnValue, webRequest);
        return null;
    }
    else if (AnnotationUtils.findAnnotation(handlerMethod, ResponseBody.class) != null) {
        handleResponseBody(returnValue, webRequest);
        return null;
    }
    else if (returnValue instanceof ModelAndView) {
        ModelAndView mav = (ModelAndView) returnValue;
        mav.getModelMap().mergeAttributes(implicitModel);
        return mav;
    }
    else if (returnValue instanceof Model) {
        return new ModelAndView().addAllObjects(implicitModel).addAllObjects(((Model) returnValue).asMap());
    }
    else if (returnValue instanceof View) {
        return new ModelAndView((View) returnValue).addAllObjects(implicitModel);
    }
    else if (AnnotationUtils.findAnnotation(handlerMethod, ModelAttribute.class) != null) {
        addReturnValueAsModelAttribute(handlerMethod, handlerType, returnValue, implicitModel);
        return new ModelAndView().addAllObjects(implicitModel);
    }
    else if (returnValue instanceof Map) {
        return new ModelAndView().addAllObjects(implicitModel).addAllObjects((Map<String, ?>) returnValue);
    }
    else if (returnValue instanceof String) {
        return new ModelAndView((String) returnValue).addAllObjects(implicitModel);
    }
    else if (returnValue == null) {
        // Either returned null or was ‘void‘ return.
        if (this.responseArgumentUsed || webRequest.isNotModified()) {
            return null;
        }
        else {
            // Assuming view name translation...
            return new ModelAndView().addAllObjects(implicitModel);
        }
    }
    else if (!BeanUtils.isSimpleProperty(returnValue.getClass())) {
        // Assume a single model attribute...
        addReturnValueAsModelAttribute(handlerMethod, handlerType, returnValue, implicitModel);
        return new ModelAndView().addAllObjects(implicitModel);
    }
    else {
        throw new IllegalArgumentException("Invalid handler method return value: " + returnValue);
    }
}

这个方法很重要,它描述的是将 handler 方法返回值解析为 ModelAndView 的过程,从中可以看出返回值可以为很多类型。建议大家都看看。这篇文章不对具体的每个返回值类型进行说明。

3.从官方描述中,也可以看出 ModelAndView 是分为两部分的,Model 和 View。这里就分两部分来说明。其中 View 部分其实是视图渲染问题。

4.Model 模型。

这里所说的 Model ,不单单指的就是 Model 具体这个类。而是描述的 SpringMVC 如何将 数据存放到 Model 中,以便在目标页面中使用。

(1)怎么存放

上面部分已经说过,通过 ModelAndView 对象来处理返回值问题。那么就可以通过如下的方式向 Model 中存入数据。如:

@RequestMapping("/testModelAndView")
public ModelAndView testModelAndView() {
  ModelAndView mv = new ModelAndView();
  mv.setViewName("success");
  mv.addObject("testKey", "testValue");
  return mv;
}

通过 ModelAndView 对象的 addObject() 方法 可以向模型中添加数据。

事实上,可以在方法的入参处添加 Model 类型 或 Map 类型的参数,可以通过向其中添加数据来完成向模型中数据的添加。如:

@RequestMapping("/testModelAndView02")
public String testModelAndView2(Map map) {
  map.put("testKey", "testValue");
  return "success";
}

为什么向 handler 方法的入参处添加 Model 类型或 Map 类型参数添加数据,就能完成 模型数据的添加。

通过断点发现传入目标方法的 Map 实际类型是 BindingAwareModelMap 这个类型。

源码解析:

org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter#invokeHandlerMethod() 方法中:

ExtendedModelMap implicitModel = new BindingAwareModelMap(); // 在这里创建的
Object result = methodInvoker.invokeHandlerMethod(handlerMethod, handler, webRequest, implicitModel);

org.springframework.web.bind.annotation.support.HandlerMethodInvoker#resolveHandlerArguments() 方法中

if (Model.class.isAssignableFrom(paramType) || Map.class.isAssignableFrom(paramType)) {
    if (!paramType.isAssignableFrom(implicitModel.getClass())) {
        throw new IllegalStateException("Argument [" + paramType.getSimpleName() + "] is of type " +
                "Model or Map but is not assignable from the actual model. You may need to switch " +
                "newer MVC infrastructure classes to use this argument.");
    }
    args[i] = implicitModel;//然后在这个地方进行的赋值。
}

可以看出在目标方法处的 所有 Map 类型(包括其子类型)或是 Model 类型(包括其子类型)的参数都会被转换为 BindingAwareModelMap 这个类型。

这里对 BindingAwareModelMap 类型进行说明:

(2)存放什么

从上面可以看出,存放的是键值对类型的 Map 或 Model。

(3)存放到何处

看下面这个例子:

@RequestMapping("/testModelAndView")
public ModelAndView testModelAndView() {
  ModelAndView mv = new ModelAndView();
  mv.setViewName("success");
  mv.addObject("testKey", "testValue");
  return mv;
}

success.jsp

通过测试,发现 Model 中的数据默认被存放到了 Request 域中。

5.总结

SpringMVC 通过 ModelAndView 解决了 handler 方法返回值问题,明白了 handler 方法返回值可以是何种类型,为什么说 ModelAndView 解决了 handler方法返回值问题,因为 handler 方法的

返回值最终都会被转换成 ModelAndView 对象。也详细的介绍了 Model 可以作为 handler 方法的入参使用,这里所说的 Model 也不仅仅是指 Model 这个类型,也指实现了 Model 或 Map 接口的

类型。也明白了 Model 中存放的什么,存放到了哪里。至此,SpringMVC 方法的返回值问题已经学习完。接下来要学习的是:既然返回值都已经有了,那么该如何去处理呢?——即handler 方法返回

值处理问题,也指视图渲染问题。另外,还有两个注解没有进行说明,@SessionAttribute和@ModelAttribute,仔细来说,其实他们两个注解也可以算到 Model 中,这里会再写一篇文章来单独说明

它两。

时间: 2024-10-10 22:29:03

SpringMVC基础——返回值问题的相关文章

springmvc 之 返回值

springMVC对于controller处理方法返回值的可选类型 spring mvc 支持如下的返回方式:ModelAndView, Model, ModelMap, Map,View, String, void. ModelAndView @RequestMapping("/hello") public ModelAndView helloWorld() { String message = "Hello World, Spring 3.x!"; return

SpringMVC 拦截返回值,并自定义

有关取代mvc:annotation-driven使用自定义配置请看: http://blog.csdn.net/cml_blog/article/details/45222431 1.在项目开发中,自定义全局返回值拦截是非常实用的,就如在Struts2的拦截器中,可以根据Action的返回值自定义返回信息,如果返回SUCCESS就统一返回一个成功的json对象,如果FAIL就返回全局的定义信息 2.配置xml: <context:component-scan base-package="

SpringMVC Controller 返回值的可选类型

spring mvc 支持如下的返回方式:ModelAndView, Model, ModelMap, Map,View, String, void. ModelAndView @RequestMapping("/hello") public ModelAndView helloWorld() { String message = "Hello World, Spring 3.x!"; return new ModelAndView("hello"

Springmvc+ajaxsubmit()返回值的问题

纠结了一下午,ajax提交form的时候总是不调success的函数,后来才知道是返回值类型的问题,前台定义的datatype是json, 而我后台返回的是一个String,所以调了error的函数,所以需要返回一个Json的对象 可是我只需要返回一个String,查了一下发现用下面的形式可以解决 $("#form").ajaxSubmit(function(result) {      alert(result);   });

spring mvc 第三天【注解实现springmvc Handler返回值为Object 的配置】

这里使用的是在前台发起请求Handler,后台伪造数据响应给前台, 解决方案:将之前的viewResolver抹掉,配置对应(request)请求的Handler信息如下 之前Handler返回的都直接就是逻辑视图名,并且需要配置相对应的固定的视图解析器,不太方便,这次需要返回Object,使Handler更灵活. 既然return 之后的值不再是逻辑视图名称了,那么就让@ResponseBody来帮忙吧 这里有两个新的注解: @RequestBody 将HTTP请求正文转换为适合的HttpMe

SpringMVC学习笔记三:Controller的返回值

springMVC的返回值有ModelAndView,String,void,Object类型 项目目录树: 该项目是在前面项目的基础上修改的,这里的pom.xml文件需要加入使用到的包,应为@ResponseBody需要使用的包 <dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifactId>jackson-core</artifactId> <version&

SpringMVC返回值类型及响应数据类型

1.SpringMVC 和 Struts2 的优略分析 共同点: 它们都是表现层框架,都是基于 MVC 模型编写的. 它们的底层都离不开原始 ServletAPI. 它们处理请求的机制都是一个核心控制器. 区别: Spring MVC 的入口是 Servlet, 而 Struts2 是 Filter. Spring MVC 是基于方法设计的,而 Struts2 是基于类,Struts2 每次执行都会创建一个动作类.所以 Spring MVC 会稍微比 Struts2 快些. Spring MVC

SpringMVC学习(七)——Controller类的方法返回值

本文所有案例代码的编写均建立在前文SpringMVC学习(六)——SpringMVC高级参数绑定与@RequestMapping注解的案例基础之上,因此希望读者能仔细阅读这篇文章. 返回ModelAndView Controller类方法中定义ModelAndView对象并返回,对象中可添加model数据.指定view.之前我就已讲过,在此并不过多赘述. 返回void 在Controller类方法形参上可以定义request和response,使用request或response指定响应结果:

SpringMVC中使用@ResponseBody注解返回值,Ajax取得中文乱码解决方法

Spring使用AnnotationMethodHandlerAdapter的handleResponseBody方法, AnnotationMethodHandlerAdapter使用request header中"Accept"的值和messageConverter支持的MediaType进行匹配,然后会用"Accept"的第一个值写入 response的"Content-Type".一般的请求都是通过浏览器进行的,request heade