简单分析springmvc是如何解析view视图,并返回页面给前端

SpringMVC配置视图解析器

<bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property key="prefix" value="/WEB-INF/jsp/"/>
<property key="suffix" value=".jsp" />
</bean>

配置的为jsp的解析器

ViewResolver接口

其内部只有一个接口方法,具体如下

View resolveViewName(String viewName, Locale locale) throws Exception;

由上可知其是通过解析ViewName来得到View视图对象

ViewResolver实现类-BeanNameViewResolver

其意图是参数viewName就是springmvc上下文中的beanName对象,具体源码如下

	@Override
public View resolveViewName(String viewName, Locale locale) throws BeansException {
//获得springmvc上下文
ApplicationContext context = getApplicationContext();
if (!context.containsBean(viewName)) {
//viewName不存在,则直接返回null
return null;
}
if (!context.isTypeMatch(viewName, View.class)) {
//viewName对应的beanName不是View.class的实现类,则直接返回
return null;
}
return context.getBean(viewName, View.class);
}

BeanNameResolver表示返回的viewName必须是springmvc上下文中的beanName并且对应的实体类必须是View.class的实现类,否则返回null

ViewResolver实现类-AbstractCachingViewResolver

其是我们常用解析器的抽象类,比如Freemarker/Groovy,我们直接去看其实现的接口方法,具体源码如下

	@Override
public View resolveViewName(String viewName, Locale locale) throws Exception {
//不采用缓存方案,则每次都进行创建View
if (!isCache()) {
return createView(viewName, locale);
}
else {
//此处为使用缓存情况下的获取View对象
Object cacheKey = getCacheKey(viewName, locale);
View view = this.viewAccessCache.get(cacheKey);
if (view == null) {
synchronized (this.viewCreationCache) {
view = this.viewCreationCache.get(cacheKey);
if (view == null) {
// Ask the subclass to create the View object.
view = createView(viewName, locale);
if (view == null && this.cacheUnresolved) {
//默认会返回一个null的View对象
view = UNRESOLVED_VIEW;
}
if (view != null) {
this.viewAccessCache.put(cacheKey, view);
this.viewCreationCache.put(cacheKey, view);
if (logger.isTraceEnabled()) {
logger.trace("Cached view [" + cacheKey + "]");
}
}
}
}
}
return (view != UNRESOLVED_VIEW ? view : null);
}
}

接以上代码我们接着分析AbstractCachingViewResolver#createView()方法

	protected View createView(String viewName, Locale locale) throws Exception {
//此处的loadView便是模板方法,供子类去实现
return loadView(viewName, locale);
}

此处我们只分析其某个实现类UrlBasedViewResolver

UrlBasedViewResolver-对应请求url的解析器

  1. 常用的内部属性
	public static final String REDIRECT_URL_PREFIX = "redirect:";

	public static final String FORWARD_URL_PREFIX = "forward:";
//设置前缀
private String prefix = "";
//设置后缀
private String suffix = "";
//设置指定的viewName集合
private String[] viewNames ;
  1. createView()复写父类方法,即在创建view对象前做下跳转的请求检查
	@Override
protected View createView(String viewName, Locale locale) throws Exception {
//viewNames集合为null或者对应的viewName在viewNames集合内则返回true
if (!canHandle(viewName, locale)) {
return null;
}
// Check for special "redirect:" prefix.
//检查handler返回的值为string类型时是否包含"redirect:"前缀
//此处处理的便是跳转请求,比如"redirect:/user/list"
if (viewName.startsWith(REDIRECT_URL_PREFIX)) {
String redirectUrl = viewName.substring(REDIRECT_URL_PREFIX.length());
RedirectView view = new RedirectView(redirectUrl, isRedirectContextRelative(), isRedirectHttp10Compatible());
return applyLifecycleMethods(viewName, view);
}
// Check for special "forward:" prefix.
//同"redirect:"请求,此处为服务端直接跳转
if (viewName.startsWith(FORWARD_URL_PREFIX)) {
String forwardUrl = viewName.substring(FORWARD_URL_PREFIX.length());
return new InternalResourceView(forwardUrl);
}
//通过父类再去调用loadView()方法,其实是本类也复写了此方法
return super.createView(viewName, locale);
}
  1. loadView()复写方法
	@Override
protected View loadView(String viewName, Locale locale) throws Exception {
//创建View对象
AbstractUrlBasedView view = buildView(viewName);
//将view对象与viewName绑定注册至springmvc上下文中
View result = applyLifecycleMethods(viewName, view);
return (view.checkResource(locale) ? result : null);
}
  1. buildView()创建View对象主逻辑
	protected AbstractUrlBasedView buildView(String viewName) throws Exception {
//获取类似FreemarkerView.class/GroovyView.class
AbstractUrlBasedView view = (AbstractUrlBasedView) BeanUtils.instantiateClass(getViewClass());
//设置view对应的资源路径,此处便可知我们设置prefix和suffix的作用
view.setUrl(getPrefix() + viewName + getSuffix()); //下面都是设置与UrlBasedViewResolver的相关内部属性
String contentType = getContentType();
if (contentType != null) {
view.setContentType(contentType);
} view.setRequestContextAttribute(getRequestContextAttribute());
view.setAttributesMap(getAttributesMap());
//是否暴露路径变量
Boolean exposePathVariables = getExposePathVariables();
if (exposePathVariables != null) {
view.setExposePathVariables(exposePathVariables);
}
//是否暴露springmvc的bean对象作为属性使用
Boolean exposeContextBeansAsAttributes = getExposeContextBeansAsAttributes();
if (exposeContextBeansAsAttributes != null) {
view.setExposeContextBeansAsAttributes(exposeContextBeansAsAttributes);
}
String[] exposedContextBeanNames = getExposedContextBeanNames();
if (exposedContextBeanNames != null) {
view.setExposedContextBeanNames(exposedContextBeanNames);
} return view;
}

主要作用是设置prefix和suffix参数以及对应的内部属性,可自行查阅,并通过buildView()方法创建ViewClass属性指定的View对象

AbstractTemplateViewResolver-UrlBasedViewResolver子类

添加另外的属性

  1. 内部属性
	//是否暴露request对象的attributes属性给前端引擎
private boolean exposeRequestAttributes = false;
//是否允许请求处理过程中复写request对象的attributes
private boolean allowRequestOverride = false;
//是否暴露session对象的attributes属性给前端引擎
private boolean exposeSessionAttributes = false;
//是否允许请求处理过程中复写session对象的attributes
private boolean allowSessionOverride = false;
//是否使用暴露springMacroRequestContext属性
private boolean exposeSpringMacroHelpers = true;
  1. buildView()复写方法,主要是额外添加以上的属性
	@Override
protected AbstractUrlBasedView buildView(String viewName) throws Exception {
//先调用父类的创建方法创建View对象
AbstractTemplateView view = (AbstractTemplateView) super.buildView(viewName);
//设置AbstractTemplateView的内部属性
view.setExposeRequestAttributes(this.exposeRequestAttributes);
view.setAllowRequestOverride(this.allowRequestOverride);
view.setExposeSessionAttributes(this.exposeSessionAttributes);
view.setAllowSessionOverride(this.allowSessionOverride);
view.setExposeSpringMacroHelpers(this.exposeSpringMacroHelpers);
return view;
}

在UrlBasedViewResolver的基础上设置额外的属性,属性集合见上文

FreeMarkerViewResolver-AbstractTemplateViewResolver子类

为方便理解,我们选取常用的模板引擎Freemarker,其他的引擎工具则供读者自行分析

  1. 构造函数-设置viewClass,满足上述的模板方法的viewClass的获取
	public FreeMarkerViewResolver() {
//设置的为FreemarkView.class
setViewClass(requiredViewClass());
}

小结

本节只解析了ViewResolver的简单逻辑,其根据配置的ViewClass属性,将配置的其他属性都设置到ViewClass对应的实例中,具体的关于视图的渲染,也就是view#render()方法我们放在下节讲解

SpringMVC源码情操陶冶-ViewResolver视图解析的更多相关文章

  1. SpringMVC源码情操陶冶-AnnotationDrivenBeanDefinitionParser注解解析器

    mvc:annotation-driven节点的解析器,是springmvc的核心解析器 官方注释 Open Declaration org.springframework.web.servlet.c ...

  2. SpringMVC源码情操陶冶-View视图渲染

    本节简单分析View视图对象的render方法 View接口 最重要的就是render()方法,具体源码如下 /** * Render the view given the specified mod ...

  3. SpringMVC源码情操陶冶-FreeMarker之web配置

    前言:本文不讲解FreeMarkerView视图的相关配置,其配置基本由FreeMarkerViewResolver实现,具体可参考>>>SpringMVC源码情操陶冶-ViewRe ...

  4. SpringMVC源码情操陶冶-DispatcherServlet简析(二)

    承接前文SpringMVC源码情操陶冶-DispatcherServlet类简析(一),主要讲述初始化的操作,本文将简单介绍springmvc如何处理请求 DispatcherServlet#doDi ...

  5. SpringMVC源码情操陶冶-RequestMappingHandlerAdapter适配器

    承接前文SpringMVC源码情操陶冶-HandlerAdapter适配器简析.RequestMappingHandlerAdapter适配器组件是专门处理RequestMappingHandlerM ...

  6. SpringMVC源码情操陶冶-DispatcherServlet

    本文对springmvc核心类DispatcherServlet作下简单的向导,方便博主与读者查阅 DispatcherServlet-继承关系 分析DispatcherServlet的继承关系以及主 ...

  7. SpringMVC源码情操陶冶-AbstractHandlerMethodMapping

    承接前文SpringMVC源码情操陶冶-AbstractHandlerMapping,本文将介绍如何注册HandlerMethod对象作为handler 类结构瞧一瞧 public abstract ...

  8. SpringMVC源码情操陶冶-DispatcherServlet类简析(一)

    阅读源码有利于陶冶情操,此文承接前文SpringMVC源码情操陶冶-DispatcherServlet父类简析 注意:springmvc初始化其他内容,其对应的配置文件已被加载至beanFactory ...

  9. SpringMVC源码情操陶冶-HandlerAdapter适配器简析

    springmvc中对业务的具体处理是通过HandlerAdapter适配器操作的 HandlerAdapter接口方法 列表如下 /** * Given a handler instance, re ...

随机推荐

  1. python基础操作_元组_字典操作

    #元组'''元组是不可变的列表,不能改.取值和列表一样'''tp=(1,2,3)tp1=('127.0.0.1','3307')#元组只有count 和index两个方法.lis=['127.0.0. ...

  2. [原创] 利用前端+php批量生成html文件,传入新文本,输出新的html文件

    本人因为要想自己写个小说网站练练手,在其中遇到的一些问题,将其解决方法总结出来,例如: 1:小说网站存储了大量的小说,每个小说主页都很相似,url不同,不是使用的history属性改写的,所以如果人工 ...

  3. js获取网页请求类型是http还是https

    代码如下,即可判断 var ishttps = 'https:' == document.location.protocol ? true : false; if(ishttps) { alert(& ...

  4. Web设计思想——渐进增强

    最近在拜读一本Web体验相关的书<渐进增强--跨平台用户体验设计 >,阅读后做些总结,消化一下书中的精髓. 在阅读本文前,可以先思考下面几个问题. 1. 浏览网页的目的是什么? 2. 浏览 ...

  5. CSS实现两端对齐效果

    CSS实现两端对齐效果 两端对齐,从概念上来说,其实不难理解.如果不明白什么叫两端对齐,可以玩玩word等办公软件. 下面谈谈如何实现文本的两端对齐.我所知道的大概有以下几种方法 text-align ...

  6. SSH连接不上CentOS 主机配置文件导致的原因的解决方法

    一.CentOS之SSH的安装与配置 SSH 为 Secure Shell 的缩写,由 IETF 的网络工作小组(Network Working Group)所制定SSH 为建立在应用层和传输层基础上 ...

  7. loadrunner 手工参数拼接与l oadrunner的url编码

    Acction() { //演示需要的一些变量,提前声明 char *name = "yezi_zh"; "; char *work = "engin" ...

  8. Log4j中配置日志文件相对路径

    方法一. 解决的办法自然是用相对路径代替绝对路径,其实log4j的FileAppender本身就有这样的机制,如:log4j.appender.logfile.File=${WORKDIR}/logs ...

  9. cisco模拟器之------交换机、路由器、vlan的综合实例

    主要实现功能:a)位于路由器同一侧的不同网段的主机之间实现通信. b)  位于不同路由器的主机之间实现通信. 网络拓扑图: 命令配置: switch0的配置: Switch(config)#vlan ...

  10. 【Android Developers Training】 90. 序言:解决云储存冲突

    注:本文翻译自Google官方的Android Developers Training文档,译者技术一般,由于喜爱安卓而产生了翻译的念头,纯属个人兴趣爱好. 原文链接:http://developer ...