黑马程序员————类加载器

本文详细介绍了Java类加载器的工作原理及其委托机制。包括系统默认的三个类加载器:BootStrap、ExtClassLoader、AppClassLoader的功能及加载路径。此外,还提供了自定义类加载器的实现示例。

------- android培训java培训、期待与您交流! ----------


一、类加载器:

负责加载硬盘上的java类,加载到虚拟机中。


系统中可以安装多个类的加载器。系统默认的有三个主要的类加载器,每个类负责加载特定位置的类:

三个系统默认的类加载器:BootStrap、ExtClassLoader、AppClassLoader.

BootStrap:负责加载   JRE/lib/re.jar

ExtClassLoader  负责加载 JRE/lib/ext/   目录下的jar包。

AppClassLoader  负责加载CLASSPATH指定的所有jar或目录。


同样用户可以自定义自己的类加载器,自定义是需要继承一个类

类加载器的委托机制:

当需要使用类加载器是需要谁去加载呢?

1)、首先当前线程的类加载器去加载线程中的第一个类。

2)、如果类A中引用了B,java虚拟机将会用加载类A的类加载器来加载类B

3)、还可以直接调用ClassLoader.loadClass()方法来指定某个类加载器去加载某个类


每个类加载加载类时,又先委托给其上级类加载器。即;当调用一个类加载器的时候,这个类加载器先不加载类,而是交给上一级类加载器加载,依次类推。   

这样做的目的: 实现类的集中管理,防止类的重复加载。     

自定义类加载器:

package com.heima.classloader; import java.io.ByteArrayOutputStream; import java.io.FileInputStream; import java.io.IOException; public class MyclassLaoder extends ClassLoader { //定义一个用于接收文件目录的变量; private String Classdir; public MyclassLaoder(String Classdir) { this.Classdir =Classdir; } @Override protected Class<?> findClass(String name) throws ClassNotFoundException { System.out.println("自定义的类加载器被调用"); //获取.class文件的文件名,用于加载文件 String mname = name.substring(name.lastIndexOf(".")+1,name.length()); byte[] byten = loadClassdata(mname); Class<?> clazz =defineClass(name,byten, 0, byten.length); return clazz ; } /** * 通过输入的名称调用和路径加载字节码文件 * @param name .class文件的文件名称。 * @return byte数组 */ private byte[] loadClassdata(String name) { //拼接字节码文件的位置 String classfullname = Classdir+"\\"+name+".class"; //定义数组输出流 ByteArrayOutputStream bos =null; FileInputStream fis = null; try { byte[] buf = new byte[1024]; fis = new FileInputStream(classfullname); bos = new ByteArrayOutputStream(); //定义一个偏移量的计数器 int i = -1; while((i=fis.read(buf))!=-1){ bos.write(buf, 0, i); } } catch (Exception e) { e.printStackTrace(); }finally{ //关闭输入流 if(fis!=null){ try { fis.close(); } catch (IOException e) { e.printStackTrace(); } } } return bos.toByteArray(); } }

定义一个用于使用自定义类加载器加载的类:

package com.heima.classloader; public class TestDemo { private String name ; private String addr; public String getName() { return name; } public void setName(String name) { this.name = name; } public String getAddr() { return addr; } public void setAddr(String addr) { this.addr = addr; } @Override public String toString() { return name +":"+addr; } }

 然后再一个类的main方法中使用自定义的类加载器加载:

    

package com.heima.classloader; import java.lang.reflect.Method; public class ClassLoaderDemo { /** * @param args * @throws ClassNotFoundException */ public static void main(String[] args) throws ClassNotFoundException { //创建自定义类加载器的实例对象,并且调用加载方法。 Class<?> clazz = new MyclassLaoder("E:\\workspace\\BlogCode\\classLoader").loadClass("com.heima.classloader.TestDemo"); //通过反射反射出这个字节码的每个方法,验证是否是要加载的类 Method[] Methods =clazz.getDeclaredMethods(); for(Method method :Methods) { System.out.println(method.getName()); Class[] clazzs = method.getParameterTypes(); for(Class clazz1:clazzs) { System.out.println(clazz1); } System.out.println("_______________"); } //输出加载这份字节码的加载器。 System.out.println(clazz.getClassLoader()); } }


输出结果:

自定义的类加载器被调用 toString _______________ getName _______________ setName class java.lang.String _______________ setAddr class java.lang.String _______________ getAddr _______________ com.heima.classloader.MyclassLaoder@2b39d891

自定义一个用于解密的类加载器,使用简单的加密方法。 

   类加载器中一个高级问题的实验分析:

   将自定义的Servlet类放到ext目录下,在加载的时候报找不到HttpServlet的错误。

   将Servlet.jar也放到ext目录下问题解决.

   问题分析,当把自定义的Servlet放到ext目录下的时候,加载使用的加载器是ExtClassLoader.

   而当A类引用B时,用加载A的类加载器加载B.

   所以 ExtClassLoader和他的父类都无法加载  Servlet类。只能放到ext中让ExtClassLoader加载。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值