文档章节

I.Spring源码分析之web入口

maskleo
 maskleo
发布于 2015/10/13 22:20
字数 775
阅读 28
收藏 0

web.xml

<?xml version="1.0" encoding="UTF-8"?>
<web-app version="2.4" xmlns="http://java.sun.com/xml/ns/j2ee"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee 
	http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd">
	<display-name>Archetype Created Web Application</display-name>
	<!-- 初始化 -->
	<context-param>
		<param-name>contextConfigLocation</param-name>
		<param-value>classpath:applicationContext.xml</param-value>
	</context-param>
	<!-- 监听器 -->
	<listener>
        <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
    </listener>
	<!-- 定义struts2的核心filter -->
	<filter>
		<filter-name>struts2</filter-name>
		<filter-class>
			org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter
		</filter-class>
	</filter>
	<!-- 让struts定义的核心filter拦截所有请求 -->
	<filter-mapping>
		<filter-name>struts2</filter-name>
		<url-pattern>*.do</url-pattern>
	</filter-mapping>

	<filter>
		<filter-name>encodingFilter</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>

	<!-- druid数据库监控 -->
	<filter>
		<filter-name>DruidWebStatFilter</filter-name>
		<filter-class>com.alibaba.druid.support.http.WebStatFilter</filter-class>
		<init-param>
			<param-name>exclusions</param-name>
			<param-value>*.js,*.gif,*.jpg,*.png,*.css,*.ico,/druid/*</param-value>
		</init-param>
	</filter>
	<filter-mapping>
		<filter-name>DruidWebStatFilter</filter-name>
		<url-pattern>/*</url-pattern>
	</filter-mapping>

	<servlet>
		<servlet-name>DruidStatView</servlet-name>
		<servlet-class>com.alibaba.druid.support.http.StatViewServlet</servlet-class>
	</servlet>
	<servlet-mapping>
		<servlet-name>DruidStatView</servlet-name>
		<url-pattern>/druid/*</url-pattern>
	</servlet-mapping>

	<!-- 项目欢迎界面 -->
	<welcome-file-list>
		<welcome-file>index.jsp</welcome-file>
	</welcome-file-list>
	<error-page>
		<error-code>404</error-code>
		<location>/404.html</location>
	</error-page>
	<error-page>
		<error-code>500</error-code>
		<location>/500.html</location>
	</error-page>
	<error-page>
		<error-code>502</error-code>
		<location>/502.html</location>
	</error-page>
	<error-page>
		<error-code>504</error-code>
		<location>/504.html</location>
	</error-page>
</web-app>

 

Spring 的入口 org.springframework.web.context.ContextLoaderListener

这个类继承了  org.springframework.web.context.ContextLoader

实现了javax.servlet.ServletContextListener接口

在ContextLoader中有一个静态语句块

static {
		// Load default strategy implementations from properties file.
		// This is currently strictly internal and not meant to be customized
		// by application developers.
		try {
			ClassPathResource resource = new ClassPathResource(DEFAULT_STRATEGIES_PATH, ContextLoader.class);
			defaultStrategies = PropertiesLoaderUtils.loadProperties(resource);
		}
		catch (IOException ex) {
			throw new IllegalStateException("Could not load 'ContextLoader.properties': " + ex.getMessage());
		}
	}

注意这个private static final String DEFAULT_STRATEGIES_PATH = "ContextLoader.properties";

我找了一下这个文件

是在ContextLoader.class一个目录下

来看看这个文件

# Default WebApplicationContext implementation class for ContextLoader.
# Used as fallback when no explicit context implementation has been specified as context-param.
# Not meant to be customized by application developers.

org.springframework.web.context.WebApplicationContext=org.springframework.web.context.support.XmlWebApplicationContext

 

注释:默认的WebApplicationContext接口的实现类

在上面的静态语句块的作用是把这个文件读到Properties(defaultStrategies)对象里

ContextLoaderListener实现了ServletContextListener接口 

那么启动服务的入口是这个方法contextInitialized

ContextLoader#initWebApplicationContext(ServletContext servletContext)

--》configureAndRefreshWebApplicationContext(cwac, servletContext);

注意看这一行代码String configLocationParam = sc.getInitParameter(CONFIG_LOCATION_PARAM);

public static final String CONFIG_LOCATION_PARAM = "contextConfigLocation";

这个就是获取web.xml中的内容

<context-param>
		<param-name>contextConfigLocation</param-name>
		<param-value>classpath:applicationContext.xml</param-value>
	</context-param>

获取配置文件的内容 找到spring文件的路径

接下来看wac.setConfigLocation(configLocationParam);

我们直接进这2个方法发现了一个严重的问题   2个方法的实现里面都是直接抛出异常的!!!

说明这个是被禁止调用的?

我再看了下ConfigurableWebApplicationContext    这是一个接口

再回来看到了ConfigurableWebApplicationContext cwac = (ConfigurableWebApplicationContext) this.context;  这个

接着向上找  this.context = createWebApplicationContext(servletContext);

也就是我们找到了创建 wac具体实例的地方

Class<?> contextClass = determineContextClass(sc);
protected Class<?> determineContextClass(ServletContext servletContext) {
		String contextClassName = servletContext.getInitParameter(CONTEXT_CLASS_PARAM);
		if (contextClassName != null) {
			try {
				return ClassUtils.forName(contextClassName, ClassUtils.getDefaultClassLoader());
			}
			catch (ClassNotFoundException ex) {
				throw new ApplicationContextException(
						"Failed to load custom context class [" + contextClassName + "]", ex);
			}
		}
		else {
			contextClassName = defaultStrategies.getProperty(WebApplicationContext.class.getName());
			try {
				return ClassUtils.forName(contextClassName, ContextLoader.class.getClassLoader());
			}
			catch (ClassNotFoundException ex) {
				throw new ApplicationContextException(
						"Failed to load default context class [" + contextClassName + "]", ex);
			}
		}
	}

最重要的是这一行代码:

contextClassName = defaultStrategies.getProperty(WebApplicationContext.class.getName());

我们上面提到在静态语句块里面解析了ContextLoader.properties

得到了

org.springframework.web.context.WebApplicationContext=org.springframework.web.context.support.XmlWebApplicationContext

根据代码我们知道XmlWebApplicationContext就是

wac.setConfigLocation(configLocationParam);

中wac的实例

果然我们在它的父类的父类(AbstractRefreshableConfigApplicationContext)中找到了setConfigLocation方法的实现

看里面的实现是把路径解析到configLocations中

AbstractApplicationContext#refresh()

开始加载解析和加载相关的文件

© 著作权归作者所有

共有 人打赏支持
maskleo
粉丝 18
博文 128
码字总数 17344
作品 0
深圳
程序员
私信 提问
react native转web方案:react-native-web

本文将从三个方面分享 react native 转 web 方案:react-native-web react-native-web 的使用 react-native-web 源码分析 react-native-web 实践 react-native-web:github.com/necolas/rea…......

hujiao
2018/08/20
0
0
React Native转web方案:react-native-web

原文地址:github.com/HuJiaoHJ/bl… React源码 React16源码之React Fiber架构 从源码看React异常处理 从源码看React.PureComponent 实践总结 web移动端布局的那些事儿 React Native转web方案...

大灰狼的小绵羊哥哥
2018/11/13
0
0
yii2源码分析之执行基本流程

用yii2框架用了将近2年,一直都没有去看过它底层源码, 马上快不用了,最近对其源码研究一番,哈哈 废话少说,上代码, 入口文件是web/index.php

china_lx1
2018/04/22
0
0
Spring Boot 2.x 启动全过程源码分析(上)入口类剖析

Spring Boot 的应用教程我们已经分享过很多了,今天来通过源码来分析下它的启动过程,探究下 Spring Boot 为什么这么简便的奥秘。 本篇基于 Spring Boot 2.0.3 版本进行分析,阅读本文需要有...

Java技术栈
2018/08/06
0
0
shiro登录流程

ShiroFilter Shiro提供了与Web集成的支持,其通过一个ShiroFilter入口来拦截需要安全控制的URL,然后进行相应的控制,ShiroFilter类似于如Strut2/SpringMVC这种web框架的前端 其是安全控制的...

嘿嘿!!
2016/10/20
0
0

没有更多内容

加载失败,请刷新页面

加载更多

JS 调用Angularjs 的方法

// 1. 获取 Controllerlet appElement = document.querySelector('[data-ng-controller=MessagesCtrl]');let scope = angular.element(appElement).scope();// 2. 调用方法scope.l......

Moks角木
39分钟前
2
0
dubbo+zookeeper与 eureka的区别

CAP CAP 原则指的是在一个分布式系统中,Consistency(一致性)、 Availability(可用性)、Partition tolerance(分区容错性),三者不可兼得 在分布式架构里, P必须有 Zookeeper保证C P 当...

群星纪元
49分钟前
4
0
云计算之边缘计算大势所趋

如果说边缘计算是公同认定的目标,那么我们看到,不同类型的厂商基于自身的特点,会从不同的起点、沿着不同的路径,向这个目标奔跑。上次参加阿里云的一次活动,看到他们将边缘计算的厂商分成...

linuxCool
53分钟前
1
0
前端通过后端传过来的'\n' ,''等字符串换行失败问题

后台推送换行符 '\n' 或 '<br/>' 等字符串到前台不会换行 详细描述 后台逻辑处理返回String字符串,其中包含\n或<br/>等换行符号,但是前端渲染时候却并没有真正的换行 也尝试了大佬的各种 ...

下次用oschina
今天
2
0
volatile能保证有序性吗?

在前面提到volatile关键字能禁止指令重排序,所以volatile能在一定程度上保证有序性。   volatile关键字禁止指令重排序有两层意思:   1)当程序执行到volatile变量的读操作或者写操作时...

无精疯
今天
2
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部