大家好,又见面了,我是你们的朋友全栈君。
先记录下:有时间整理下。
// Class.forName(“”).getResourceAsStream(name)
1 2 3 4 5 6 7 8 9 10 11 |
public class Test { public static void main(String[] args) { // 此时三个ClassLoader是同一个对象 System.out.println(Thread.currentThread().getContextClassLoader()); // 当前线程的类加载器 System.out.println(Test. class .getClassLoader()); // 当前类的类加载器 System.out.println(ClassLoader.getSystemClassLoader()); // 系统初始的类加载器 } } |
如果楼主了解过openfire应该对ClassLoader有比较深的理解。
打个简单的比方,你一个WEB程序,发布到Tomcat里面运行。
首先是执行Tomcat org.apache.catalina.startup.Bootstrap类,这时候的类加载器是ClassLoader.getSystemClassLoader()。
而我们后面的WEB程序,里面的jar、resources都是由Tomcat内部来加载的,所以你在代码中动态加载jar、资源文件的时候,首先应该是使用Thread.currentThread().getContextClassLoader()。如果你使用Test.class.getClassLoader(),可能会导致和当前线程所运行的类加载器不一致(因为Java天生的多线程)。
Test.class.getClassLoader()一般用在getResource,因为你想要获取某个资源文件的时候,这个资源文件的位置是相对固定的。
java的类加载机制(jvm规范)是委托模型,简单的说,如果一个类加载器想要加载一个类,首先它会委托给它的parent去加载,如果它的所有parent都没有成功的加载那么它才会自己亲自来,有点儿像儿子使唤老子的感觉。。jvm也拼爹啊,,,,,
在jvm中默认有三类loaer,bootstrap,ext,app,其中boot最大是爷爷,app最小是孙子,ext中间是爹。
它们有权限访问的classpath也不一样,boot是jdk或jre下面的lib目录,ext是jdk或jre的ext目录,而app是由用户指定的路径,比如用-cp参数指定的目录或jar。他们没有权力访问其他人的classpath,这样问题就来鸟,,,,可能有人会问狗司令大人闲得蛋疼啊,搞这么复杂,据说是为了安全考虑,避免用户的恶心意代码侵蚀jvm,,问题就是当bootstrap或ext想要加载用户指定classpath中的类就会失败,因为这俩货没有权限访问团app路径中的类的,,所以就搞了这么一个不伦不类的contextloader。。。。
资料:
ContextLoaderListener作用详解
(1) org.springframework.web.context.ContextLoaderListener 这个类被定义为监听器,并读取在参数contextConfigLocation中定义的xml 文件,如果不设置contextConfigLocation的初始参数则默认会读取 WEB-INF路径下的 application.xml文件,如果需要自定义了另外的xml 则可以在contextConfigLocation下定义,ContextLoaderListener会读取这些XML文件并产生 WebApplicationContext对象,然后将这个对象放置在ServletContext的属性里,这样我们只要可以得到Servlet就可以得到WebApplicationContext对象,并利用这个对象访问spring 容器管理的bean。
(2)解析ContextLoadListener的源代码
// 实现了servlet的ServletContextListener接口
public class ContextLoaderListener
implements ServletContextListener
{
private ContextLoader contextLoader;
// 这个是最重要的,利用contextLoader 来完成所有的工作
public void contextInitialized(ServletContextEvent event)
{
this.contextLoader = createContextLoader();
this.contextLoader.initWebApplicationContext(event.getServletContext());
}
protected ContextLoader createContextLoader()
{
return new ContextLoader();
}
public ContextLoader getContextLoader()
{
return this.contextLoader;
}
public void contextDestroyed(ServletContextEvent event)
{
if (this.contextLoader != null)
this.contextLoader.closeWebApplicationContext(event.getServletContext());
}
}
下面看一下ContextLoader .initWebApplicationContext方法
public WebApplicationContext initWebApplicationContext(ServletContext servletContext)
throws IllegalStateException, BeansException
{
if (servletContext.getAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE) != null) {
throw new IllegalStateException(“Cannot initialize context because there is already a root application context present – check whether you have multiple ContextLoader* definitions in your web.xml!”);
}
long startTime = System.currentTimeMillis();
if (this.logger.isInfoEnabled())
this.logger.info(“Root WebApplicationContext: initialization started”);
servletContext.log(“Loading Spring root WebApplicationContext”);
try
{
ApplicationContext parent = loadParentContext(servletContext);
// 这里得到WebApplicationContext对象 ,下面具体看一下这个方法是如何实现的
this.context = createWebApplicationContext(servletContext, parent);
// 将这个对象设置到 servletContext的属性里
servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, this.context);
if (this.logger.isInfoEnabled()) {
this.logger.info(“Using context class [” + this.context.getClass().getName() + “] for root WebApplicationContext”);
}
if (this.logger.isDebugEnabled()) {
this.logger.debug(“Published root WebApplicationContext [” + this.context + “] as ServletContext attribute with name [” + WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE + “]”);
}
if (this.logger.isInfoEnabled()) {
long elapsedTime = System.currentTimeMillis() – startTime;
this.logger.info(“Root WebApplicationContext: initialization completed in ” + elapsedTime + ” ms”);
}
return this.context;
}
protected WebApplicationContext createWebApplicationContext(ServletContext servletContext, ApplicationContext parent)
throws BeansException
{
// 返回在web.xml 中参数contextClass自定义类对应的对象这个类,这个类实现了XmlWebApplicationContext
// XmlWebApplicationContext继承了AbstractRefreshableWebApplicationContext类中定义的方法protected String[] getConfigLocations() ,这个方法默认可以加载contextConfigLocation中定义的xml 文件,如果你重写了这个方法还可以在任意地方加载你想要的xml文件。
Class contextClass = determineContextClass(servletContext);
if (!(ConfigurableWebApplicationContext.class.isAssignableFrom(contextClass))) {
throw new ApplicationContextException(“Custom context class [” + contextClass.getName() + “] is not of type ConfigurableWebApplicationContext”);
}
// 得到 WebApplicationContext
ConfigurableWebApplicationContext wac = (ConfigurableWebApplicationContext)BeanUtils.instantiateClass(contextClass);
wac.setParent(parent);
wac.setServletContext(servletContext);
String configLocation = servletContext.getInitParameter(“contextConfigLocation”);
if (configLocation != null) {
wac.setConfigLocations(StringUtils.tokenizeToStringArray(configLocation, “,; \t\n”));
}
wac.refresh();
return wac;
}
发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/134558.html原文链接:https://javaforall.cn
【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛
【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...