学习一下Java中的类加载器,这个是比较底层的东西,好好学习、理解一下。

 一、类加载器的介绍

1、类加载器:就是加载类的工具,在java程序中用到一个类,java虚拟机首先要把这个类的字节码加载到内存中来,但是通常这个类的原始信息是放在硬盘上的classpath指定的路径下,将.class文件的内容加载到硬盘上来,再对它进行一些处理,处理完的结果就是字节码,这些工作就是类加载器在做

扩展一下一个java程序的生命周期:(这个总是记不住,无语了。。。)

以Java为例:

电脑是不能直接执行Java程序的,一个.java程序要想被执行,首先需要编译器将高级的.java程序文件编译成.class字节码片段,字节码经过JVM(解释器)的处理后生成电脑可以直接执

行的机器码,至此java程序才能得以正确运行。

2、Java虚拟机中可以安装多个类加载器,系统中默认的有三个类加载器:Bootstrap、ExtClassLoader、AppClassLoader,每个类加载器加载特定位置的类

3、类加载器也是java类,因为其他是java类的类加载器本身也要被类加载器加载,显然必须有一个类加载器不是java类,这个加载器就是Bootstrap类加载器。

注意:Bootstrap类加载不是java类,它是嵌套在java虚拟机内核中的,java虚拟机内核一启动,它就已经在那里了,它是用C++语言写的一段二进制代码,它可以去加载别的类以及其他的类加载器。

         // sun.misc.Launcher$AppClassLoader
String classLoaderName1 = ClassLoaderTest.class.getClassLoader().getClass().getName();
System.out.println(classLoaderName1); // 为 null 说明System类是由Bootstrap类加载器加载的
System.out.println(System.class.getClassLoader()); // 循环理解 类加载器中的父子关系
ClassLoader loader = ClassLoaderTest.class.getClassLoader();
while(loader != null){
System.out.println(loader.getClass().getName());
loader = loader.getParent();
}
System.out.println(loader);

4、java虚拟机中的所有类加载器采用具有父子关系的属性结构进行组织,在实例化每个类加载器对象时候,需要为其指定一个父级类加载器对象,或者默认采用系统类加载器为其父级类加载器

5、类加载器的委托机制

当java虚拟机要加载一个类时,到底派出哪个类加载器去加载呢?

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

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

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

每个类加载器加载类时,又先委托给其上级类加载器

1)当所有祖宗类加载器没有加载到类时,回到发起者类加载器,还加载不了,则抛出ClassNotFoundException,不是去找发起者类加载器的儿子,因为没有getChild()方法,就算是有,那有多个儿子,应该交给哪一个呢?

 二、编写自己的类加载器

 1、自定义类加载器的原理

从jdk文档中找到的一个案例:(主要就是重写findClass()方法,加载二进制class文件,转换成字节码)

 //网络类加载器子类必须定义方法 findClass 和 loadClassData,以实现从网络加载类。下载组成该类的字节后,它应该使用方法 defineClass 来创建类实例。示例实现如下: 

      class NetworkClassLoader extends ClassLoader {
String host;
int port; public Class findClass(String name) {
byte[] b = loadClassData(name);
return defineClass(name, b, 0, b.length);
} private byte[] loadClassData(String name) {
// load the class data from the connection
. . .
}
}

2、自己定义classLoader

这个写起来真的费劲啊,听着张老师的视频讲解都感觉很费劲,这个要是自己能写出来,那真的要对class的加载机制很熟悉,看着张老师解决问题的能力,感觉自己真的还有很多东西需要学习,自己的解决问题的能力几乎是0,但是这背后肯定离不开强大的知识体系的支撑,自己的基础知识不够扎实,如何去谈解决问题,可能你都发现不了问题。

代码拿出来,不过还有很多难点,希望以后自己能够完全理解(这个重写ClassLoader应该不难,但是如何让你写的ClassLoader来使用起来,我觉得这个是比较难的,你要知道在java中的类加载机制,树状的类加载关系,只有当父类加载不到类的时候,才会轮到你自己重写的ClassLoader来加载所需要加载的类,问题是如何让父类加载不到,而让你自己的ClassLoader加载到呢?这个是问题的所在,也是解决问题的切入点

 public class MyClassLoader extends ClassLoader{

     private String classDir;

     public static void main(String[] args) throws Exception{
String srcPath = args[0];
String destDir = args[1]; String destFileName = srcPath.substring(srcPath.lastIndexOf("\\")+1);
String destPath = destDir + "\\" + destFileName; FileInputStream fis = new FileInputStream(srcPath);
FileOutputStream fos = new FileOutputStream(destPath);
cypher(fis, fos);
fis.close();
fos.close(); } private static void cypher(InputStream ips, OutputStream ops) throws Exception{
int b = -1; // 读字节流读到最后为 -1
while((b = ips.read()) != -1 ){
// 进行异或运算 就是把二进制中的0编程1 1变成0 进行简单的加密
ops.write(b ^ 0xff);
}
} @Override
protected Class<?> findClass(String name) throws ClassNotFoundException {
// 截取包名中最后的类的名称
String classFileName = classDir + "\\" + name.substring(name.lastIndexOf(".")+1) + ".class";
System.out.println("MyClassLoader中加载的类的名称:" + classFileName);
try {
FileInputStream fis = new FileInputStream(classFileName);
ByteArrayOutputStream bos = new ByteArrayOutputStream();
cypher(fis, bos);
fis.close(); byte[] bytes = bos.toByteArray();
return defineClass(bytes, 0, bytes.length);
} catch (Exception e) {
e.printStackTrace();
} return super.findClass(name);
} public MyClassLoader(){ } public MyClassLoader(String classDir){
this.classDir = classDir;
}
}

写一下测试代码:

         //父类加载的是全路径加包名的类
Class clazz = new MyClassLoader("itcastlib").loadClass("com.ssc.classLoader.ClassLoaderAttachment");
// 子类继承父类的加载的是相对路径的类 只有当父类没有找到类的时候 子类才进行加载
//Class clazz = new MyClassLoader("itcastlib").loadClass("ClassLoaderAttachment");
Date d1 = (Date) clazz.newInstance();
System.out.println(d1);

这里的让重写的ClassLoader加载指定的类的解决办法是:我把编译好的指定的类的class文件放在 itcastlib文件夹中,这个其实是上面加密后的class文件,也就是MyClassLoader类中main函数执行之后生成在  itcastlib 文件夹中的,但是父类会去E:\20181124\workspace\Test\bin\com\ssc\classLoader\ClassLoaderAttachment.class 路径下找那个类的class文件,而当我用重写的ClassLoader加载时,会从itcastlib 文件夹下去加载该类的class文件,并且我自己的ClassLoader会执行解密方法,对class文件的字节码进行解密。否则父类加载加密的class文件的时候,会报错的,因为父类中的findClass()方法并没有我的解密方法。

下图中的样子:

父类加载报错:

总结:

这样写出来,感觉自己稍微理清楚一点了。嘻嘻嘻

其实,在好多的框架中都会自己设计所需要的类加载器,但是我之前好像并不知道,看了一下Tomcat中的servlet的加载问题,看到了Tomcat中使用了WebappClassLoader、StandardClassLoader来加载servlet以及Tomcat内部实现的HttpServlet,这个以后可以往深里仔细去看一下

Java中的类加载器--Class loader的更多相关文章

  1. Java中的类加载器

    转载:http://blog.csdn.net/zhangjg_blog/article/details/16102131 从java的动态性到类加载机制   我们知道,Java是一种动态语言.那么怎 ...

  2. Java中的类加载器以及Tomcat的类加载机制

    在加载阶段,虚拟机需要完成以下三件事情: 1.通过一个类的全限定名来获取其定义的二进制字节流. 2.将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构. 3.在Java堆中生成一个代表这个类 ...

  3. [读书笔记]java中的类加载器

    以下内容大多来自周志明的<深入理解Java虚拟机>. 类加载器是java的一项创新,也是java流行的重要原因之一,它最初是为了满足java applet的需求而开发出来. 什么是appl ...

  4. java中的类加载器ClassLoader和类初始化

    每个类编译后产生一个Class对象,存储在.class文件中,JVM使用类加载器(Class Loader)来加载类的字节码文件(.class),类加载器实质上是一条类加载器链,一般的,我们只会用到一 ...

  5. 关于java中的类加载器

    什么是类加载器? 类加载器是专门负责加载类的命令或者说工具 ClassLoader java中的3个类加载器 JDK中自带了3个类加载器 启动类加载器 扩展类加载器 应用类加载器 假设有这样一段代码 ...

  6. Java中的类加载器----ClassLoader

    1.简单的讲类加载器就是加载类. 在一个类要被执行时,首先会被从硬盘中加载到内存中,这个任务就是由类加载器来完成,如果加载不成功时,类是无法被执行的.类加载器执行的都是字节码二进制文件.   帮助文档 ...

  7. JAVA基础_类加载器

    什么是类加载器 类加载器是Java语言在1.0版本就引入的.最初是为了满足JavaApplet需要.现在类加载器在Web容器和OSGI中得到了广泛的应用,一般来说,Java应用的开发人员不需要直接同类 ...

  8. 黑马程序员——【Java高新技术】——类加载器

    ---------- android培训.java培训.期待与您交流! ---------- 一.概述 (一)类加载器(class loader) 用来动态加载Java类的工具,它本身也是Java类. ...

  9. 黑马程序员:Java基础总结----类加载器

    黑马程序员:Java基础总结 类加载器   ASP.Net+Android+IO开发 . .Net培训 .期待与您交流! 类加载器 Java虚拟机中可以安装多个类加载器,系统默认三个主要类加载器,每个 ...

随机推荐

  1. E20180423-hm

    disclosure  n. (发明等的) 公开; 泄露,揭露; 开诚布公的话; 被公开的事情,被披露的秘闻; alignment n. 结盟; 队列,排成直线; 校直,调整; [工] 准线; ali ...

  2. 一个表的两个字段具有相同的类型。如何仅用SQL语句交换这两列的数据?

    --假设为A B两个字段--查询Select A As B, B As A From TableName --更新Update TableName Set A = B, B = A

  3. hdoj1097

    好久没有遇到过这样的题,似乎记得以前完全就是靠规律啊什么的.... 然后刚刚看到,这不就是快速幂取膜就好了嘛- #include <stdio.h> #include <string ...

  4. NOIp2016 蚯蚓 【二叉堆/答案单调性】By cellur925

    题目传送门 $Sol$ $50pts$:我们考虑$q==0$的情况,每次在所有的蚯蚓中找到一只长度最大的,这非常二叉堆.所以我们可以用一个优先队列,随便水一下就有50分.($NOIp$的分真这么好拿? ...

  5. java实训 :异常(try-catch执行顺序与自定义异常)

    关键字: try:执行可能产生异常的代码 catch:捕获异常 finally:无论是否发生异常代码总能执行 throws:声明方法可能要抛出的各种异常 throw:手动抛出自定义异常 用 try-c ...

  6. PTA天梯赛训练题L1-064:估值一亿的AI核心代码(字符串模拟)

    Update:smz说regex秒过Orz,yzd记在这里了. 听说今年天梯赛有个烦人的模拟,我便被队友逼着试做一下……一发15,二发20.记一记,要不然枉费我写这么久…… 自己还是代码能力太菜了,校 ...

  7. 洛谷 P3312 [SDOI2014]数表

    式子化出来是$\sum_{T=1}^m{\lfloor}\frac{n}{T}{\rfloor}{\lfloor}\frac{m}{T}{\rfloor}\sum_{k|T}\mu(\frac{T}{ ...

  8. mysql学习之通过文件创建数据库以及添加数据

    转自:http://blog.163.com/wujicaiguai@126/blog/static/170171558201411311547655/ 1.# 创建数据库语句 create data ...

  9. Python+selenium定位不到元素的问题及解决方案

    在操作过程中主要遇到两种阻塞的问题,总结如下: 1.页面中有iframe,定位元素时,需要用switch_to.frame()转换到元素所在的frame上再去定位 2.遇到一种新情况,有些按钮在htm ...

  10. 474 Ones and Zeroes 一和零

    在计算机界中,我们总是追求用有限的资源获取最大的收益.现在,假设你分别支配着 m 个 0 和 n 个 1.另外,还有一个仅包含 0 和 1 字符串的数组.你的任务是使用给定的 m 个 0 和 n 个 ...