0
点赞
收藏
分享

微信扫一扫

Spring Boot 形参Map并没有添加到类似于ModelAndView中,但是却可以页面取到相应的值?


Spring Boot 形参Map并没有添加到类似于ModelAndView中,但是却可以页面取到相应的值?_从零开始学Spring Boot


需求缘起:

       有网友留言:

感谢讲解,思路很清晰,不过有点疑惑,为什么最后结尾的时候,那个形参Map并没有添加到类似于ModelAndView中,但是页面却可以取到相应的值?

 

本节大纲:

(1)留言代码翻译

(2)问题分析

(3)Spring MVC数据模型

(4)写法延伸

 

       接下来看下具体的内容:

留言代码翻译:

我们在controller常写的代码是这样子的:

 

@RequestMapping("/index")
    public ModelAndView index(){
       ModelAndView mv = new ModelAndView("/index");
       mv.addObject("name", "[Angel -- 守护天使]");
       return mv;
    }



但是在我的好多代码里,却是这样子写的:

//http://127.0.0.1:8080/spring-boot/index2
    @RequestMapping("/index2")
    public Stringindex2(Map<String,Object> map){
       map.put("name", "[Angel -- 守护天使]");
       return "/index";
    }




第一个代码使用了ModelAndView,第二个直接使用了map,那为什么第二个例子的参数name也能够在前端获取到呢,或者说为什么使用map也可以呢?这就是本篇文章要解决的问题。

 

问题分析:

我们看下ModelAndView的代码:

public class ModelAndView {
 
    /** View instance or view name String */
    private Object view;
 
    /** Model Map */
    private ModelMap model;
 
    /** Optional HTTP status for the response */
    private HttpStatus status;
 
    /** Indicates whether or not this instance has been cleared with a call to{@link #clear()} */
    private boolean cleared = false;
    //省略其它代码…
}




       在这里有一个属性ModelMap,这个很重要,我们在看里面的addObject,源码如下:

  

public ModelAndView addObject(String attributeName, Object attributeValue) {
       getModelMap().addAttribute(attributeName, attributeValue);
       return this;
    }



       所以这里添加是使用了modelMap属性进行添加的,看下ModelMap源码:

public class ModelMap extendsLinkedHashMap<String, Object> {
 
    /**
     * Construct a new, empty {@code ModelMap}.
     */
    public ModelMap() {
    }
    //省略其它代码…
}




       可以看出ModelMap继承了LinkedHashMap,而LinkedHashMap又继承了Map,所以这里map为什么也是可以设置前端属性值的,慢慢的就清楚了。

 

Spring MVC数据模型:

SpringMVC在调用方法前会创建一个隐含的数据模型,作为模型数据的存储容器,成为”隐含模型”。如果处理方法入参为Map或者Model类型,SpringMVC会将隐含模型的引用传递给这些入参。

       spring Web MVC 提供Model、Map或ModelMap让我们能去暴露渲染视图需要的模型数据。

       看如下的一段很有趣的代码:

 

@RequestMapping(value = "/index3")
    public String index3(Model model,Map<String,Object> model2, ModelMap model3) {
        model.addAttribute("a", "a");
        model2.put("b", "b");
        model3.put("c", "c");
        System.out.println(model == model2);
        System.out.println(model2 == model3);
        return "index3";
     }



       控制台的打印是:

true

true

       虽然此处注入的是三个不同的类型(Model model, Map model2,ModelMap model3),但三者是同一个对象。

       AnnotationMethodHandlerAdapter和RequestMappingHandlerAdapter将使用BindingAwareModelMap作为模型对象的实现,即此处我们的形参(Modelmodel, Map model2, ModelMap model3)都是同一个BindingAwareModelMap实例。

       我们跟踪源代码可以发现:

BindingAwareModelMapextends ExtendedModelMap

而ExtendedModelMap:

ExtendedModelMap extendsModelMap implements Model

       所以到头来这些都是一个引用,就可以解释的通了。

 

ModelAndView特别说明:

       我们会发现上面并没有过多的提到这个,ModelAndView:是包含ModelMap 和视图对象的容器。正如名字暗示的一样既包含模型也包含视图,而ModelMap只是包含模型的信息。

       一旦你知道这些的话,这些涉及到的类就可以比较好的理解了。

 

写法延伸:

       从上面的讲解中,我们在controller可以有很多种写法,接着往下看:

(1)写法1:String写法

@RequestMapping(value = "/index4")
    public String index4() {
        return "index4";
     }




 

(2)写法2:String加Map写法

  

@RequestMapping(value = "/index5")
    public Stringindex5(Map<String,Object> map) {
       map.put("name", "Andy");
        return "index5";
     }



 

(3)写法3:String加ModelMap写法

 

@RequestMapping(value = "/index6")
    public String index6(ModelMap modelMap) {
       modelMap.addAttribute("name","Andy-2"); 
        return "index5";
     }



 

(4)写法4:String加接口Model写法

@RequestMapping(value = "/index7")
    public String index7(Model model) {
       model.addAttribute("name","Andy-3"); 
        return "index5";
     }




 

(5)写法5:String加大杂烩写法

@RequestMapping(value = "/index3")
    public String index3(Model model,Map<String,Object> model2, ModelMap model3) {
        model.addAttribute("a", "a");
        model2.put("b", "b");
        model3.put("c", "c");
        return "index3";
     }




 

(6)写法6:ModelAndView写法

   

@RequestMapping("/index")
    public ModelAndView index(){
       ModelAndView mv = new ModelAndView("/index");
       mv.addObject("name", "[Angel -- 守护天使]");
       return mv;
    }



       好了,可能还有别的写法,就到这里吧。那么这多的写法,我们用哪一种呢?所以第一一个项目要稍微保持统一,选择一到二两种就可以了,博主比较喜欢的就是Map和ModelAndView这两种方式了。

 

 

举报
0 条评论