ClassLoader简介

作用:ClassLoader是用来动态加载class文件到内存中。

JAVA默认提供三个classLoader
  1. BootStrap ClassLoader:启动类加载器,负责加载JDK中的核心类库。
  2. Extension ClassLoader:扩展类加载器,负责加载JAVA的扩展类库。
  3. App ClassLoader: 系统类加载器, 加载应用程序classpath目录下的所有jar和class文件。
  4. 自定义类加载器
ClassLoader类加载的原理

原理介绍:
ClassLoader使用双亲委托来搜索类,每一个ClassLoader实例都有一个父类加载器的引用。
当一个ClassLoader实例需要加载某个类时,会在亲自搜索这个类之前,先把任务委托给父类加载器,这个过程时由上至下的。
首先由最顶层的类加载器BootStrap ClassLoader试图加载,如果没有加载到,则把任务转交给Extension ClassLoader加载。
如果也没加载到,则转交给App ClassLoader进行加载。
如果也没加载到,则返回给委托的发起者,由它到指定的文件系统或者网络等URL中加载该类。
如果都没有加载到这个类,就抛出ClassNotFoundException异常。
否则将这个找到的类生成一个类的定义,并将它加载到内存中,最后返回这个类在内存中的Class实例对象。

使用原因:
可以避免重复加载,因为当父类加载了该类的时候,子ClassLoader就没有必要重新加载一次。
如果不使用这种委托模式,当用户可以使用自定义String来动态替换java核心api中定义的类型。
而双亲委托的方式可以避免这种情况,因为String在启动时已经被BootStrap ClassLoader加载,用户自定义的ClassLoader永远无法加载一个自己写的String,除非改变JDK中ClassLoader搜索类的默认算法。

JVM在搜索类的时候是如何判断两个Class是相同的?

不仅需要判断两个类的类名是否相同,而且要判断是否由同一个类加载器实例加载。只有同时满足两种情况,JVM才认为两个Class是相同的。
就算两个class是同一份class字节码,如果被两个不同的ClassLoader实例加载,JVM也会认为是两个不同的class。

ClassLoader的体系架构:
这里写图片描述

定义自己的ClassLoader

JAVA提供了默认的ClassLoader,为什么要定义自己的类加载器?

因为JAVA提供的默认ClassLoader只加载指定目录下的jar和Class,如果想加载其他位置的类或jar时,就需要定义自己的ClassLoader。目前的web服务器中都定义了自己的类加载器,用于加载web应用制定目录下的类库(jar或class),入weblogic、Jboss、tomcat等。

例子:
1、新建一个web工程httpweb
2、新建一个ClassLoaderServletTest,用于打印web容器中的ClassLoader层次结构

import java.io.IOException; 
import java.io.PrintWriter;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class ClassLoaderServletTest extends HttpServlet {  
    public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {  
        PrintWriter out = response.getWriter(); 
        ClassLoader loader = this.getClass().getClassLoader();          
        while(loader != null) {  
            loader = loader.getParent();  
        }  
         out.write(String.valueOf(loader));  
         out.close(); 
    }

    public void doPost(HttpServletRequest request, HttpServletResponse response) {
        this.doGet(request, response);  
    }  
}

3、配置Servlet,并启动服务

<?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">  
<servlet>  
    <servlet-name>ClassLoaderServletTest</servlet-name> 
     <servlet-class>ClassLoaderServletTest</servlet-class>  
      <servlet-mapping> 
        <url-pattern>/servlet/ClassLoaderServletTest</url-pattern> 
     </servlet-mapping>  
   <welcome-file>index.jsp</welcome-file> 
</web-app>  

4、访问Servlet,获得显示结果
这里写图片描述

定义自己的类加载器分为两步:
1.继承java.lang.ClassLoader
2.重写父类的findClass方法
当JDK中的loadClass方法搜索不到类时,loadClass方法就会调用findClass方法搜索类,所以只需要重写该方法即可。

参考:https://blog.csdn.net/briblue/article/details/54973413

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值