1.类加载器及委托机制的深入分析

  > 类加载器的作用:一个java文件中的出现的类,首先要把这个类的字节码加载到内存中,这个类的信息放在硬盘的classPath下的class文件中,

    把class文件中的内容加载到内存中去,在进行一些处理,处理完的结果就是字节码,这一系列工作是类加载器在做。

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

      BootStrap,ExtClassLoader, AppClassLoader

  > 类加载器也是Java类,因为其他是java类的类加载器本身也要被类加载器加载,显然必须有第一个类加载器不是java类,

   这正是BootStrap

   BootStrap 很特殊,不是一个java类,是嵌套在jvm内核里的,jvm内核一启动BootStrap就已经在那里边了,BootStrap是由C++语言写的一段二进制代码,可以去加载别的类,包括别的类加载器

    System类是由BootStrap类加载器加载

用eclipse 的打包工具将ClassLoaderTest 输出成jre/lib/ext 目录下的itcast.jar包,再在eclipse中运行这个类,运行结果显示为ExtClassLoader。此时的环境状态时classpath目录有ClassLoaderTest.class, ext/itcast.jar 包中也有ClassLoaderTest.class。

当ExtClassLoader中能找到时,使用的是ExtClassLoader,当ExtClassLoader中找不到时,会去AppClassLoader中找,这里的原理是类加载器的委托机制。

public static void main(String[] args) {
System.out.println(ClassLoaderTest.class.getClassLoader().getClass()
.getName()); //sun.misc.Launcher$AppClassLoader /**
* System 的类加载器为null 不代表没有类加载器,代表是由一个特殊的类加载器加载 BootStrap类加载器
* 因为是嵌套在jvm里的,用java获取肯定获取不到,所有为null
*/
System.out.println(System.class.getClassLoader()); //null /**
* 以下程序打印
* sun.misc.Launcher$AppClassLoader
sun.misc.Launcher$ExtClassLoader
null
*/
ClassLoader loader = ClassLoaderTest.class.getClassLoader();
while(loader != null){
System.out.println(loader.getClass().getName());
loader = loader.getParent();
}
System.out.println(loader);
}

sun.misc.Launcher$ExtClassLoader

2.类加载器的委托机制

  2.1.可以定义类加载器,自定义类加载器需要继承ClassLoader,自定义的类加载器可以指定一个特定的目录

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

    > 首先当前线程的类加载器去加载线程中的第一个类。getContextClassLoader() setContextClassLoader(ClassLoader cl)

    > 如果类A 中引用了类B ,Java虚拟机将使用加载类A的类加载器来加载类B。

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

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

    > 当所有祖宗类加载器没有加载到类,回到发起者类加载器,还加载不了,则抛ClassNotFoundException,

      不是再去找发起者类加载器的儿子,因为没有getChild方法,即使有,那有多少个儿子呢,找哪一个呢?

    > 对着类加载器的层次结构图和委托加载原理,解释先前将ClassLoaderTest输出成jar/lib/ext 目录下

      的itcast.jar包中后,运行结果为ExtClassLoader的原因

  2.4 模板方法设计模式

    父类 -->loadClass/findClass

      |----子类1(自己干)

      |----子类2(自己干)

    每个子类都有自己的方法逻辑,每个子类加载时都会去找其父节点,这个流程每个子类都会有,所以这个代码放在父类中

    我们要覆盖的是findClass,如果覆盖loadClass就会破坏这个流程

    得到class文件的转换成字节码 --> definedClass()

    

protected Class<?> defineClass(String name, byte[] b, int off, int len)

  Api中的一段代码:

The network class loader subclass must define the methods findClass and loadClassData to load a class from the network. Once it has downloaded the bytes that make up the class, it should use the method defineClass to create a class instance. A sample implementation is: 

     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
. . .
}
}

编写对class文件进行加密的工具类

  

public class MyclassLoader {

    public static void main(String[] args) throws Exception {
String srcPath = args[0];
String destDir = args[1];
FileInputStream fis = new FileInputStream(srcPath);
String destFileName = srcPath.substring(srcPath.lastIndexOf('\\')+1);
String destPath = destDir + "\\" + destFileName;
FileOutputStream fos = new FileOutputStream(destPath);
cypher(fis, fos);
fos.close();
fis.close();
} /**
* 加密
* @param is
* @param ot
* @throws IOException
*/
private static void cypher(InputStream is,OutputStream ot) throws IOException{
int b = -1;
while((b = is.read()) != -1){
ot.write(b ^ 0xff); //异或 0变1 1变0
}
}
}

在工程目录下新建一个文件夹 itcastlib

点击鼠标右键-> Run Configuration  填入运行参数

运行后,在文件夹itcastlib中新增一个class文件

  有包名的类不能调用无包名的类

  使用正常的class文件能够正常打印

  System.out.println(new ClassLoaderAttachment().toString());

  使用加密后的class文件替换正常的没有加密的class文件,并打印toString()

  则会抛出异常

 必须用自定义的类加载器,对加密的class可以进行解密的操作

private String classDir;

    @Override
protected Class<?> findClass(String name) throws ClassNotFoundException {
String classFileName = classDir + "\\" + name + ".class";
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) {
}
return null;
} public MyclassLoader(String classDir){
this.classDir = classDir;
}

Class clazz = new MyclassLoader("itcastlib").loadClass("ClassLoaderAttachment");
Date d1 = (Date)clazz.newInstance(); //ClassLoaderAttachment这个类不能显示出现,因为这个类还是乱的
System.out.println(d1); //hello,itcast 解密成功

java高新技术-类加载器的更多相关文章

  1. java自定义类加载器

    前言 java反射,最常用的Class.forName()方法.做毕设的时候,接收到代码字符串,通过 JavaCompiler将代码字符串生成A.class文件(存放在classpath下,也就是ec ...

  2. Java虚拟机类加载器及双亲委派机制

    所谓的类加载器(Class Loader)就是加载Java类到Java虚拟机中的,前面<面试官,不要再问我"Java虚拟机类加载机制"了>中已经介绍了具体加载class ...

  3. 深入探讨java的类加载器

    类加载器是 Java 语言的一个创新,也是 Java 语言流行的重要原因之一.它使得 Java 类可以被动态加载到 Java 虚拟机中并执行.类加载器从 JDK 1.0 就出现了,最初是为了满足 Ja ...

  4. java 中类加载器

    jar 运行过程和类加载机制有关,而类加载机制又和我们自定义的类加载器有关,现在我们先来了解一下双亲委派模式. java 中类加载器分为三个: BootstrapClassLoader 负责加载 ${ ...

  5. 深入理解Java虚拟机 - 类加载器

    引子:       类加载器(classloader)是独立于虚拟机之外,可以独立实现的代码模块.     OSGi使用了类加载器的这一特点实现其热插拔的特性       Java同C++等语言不通, ...

  6. 分析Java的类加载器与ClassLoader(二):classpath与查找类字节码的顺序,分析ExtClassLoader与AppClassLoader的源码

    先回顾一下classpath classpath的作用: classpath的作用是指定查找类的路径:当使用java命令执行一个类(类中的main方法)时,会从classpath中进行查找这个类. 指 ...

  7. java面向对象--类加载器及Class对象

    类加载器 jvm 和 类的关系 当调用 java命令运行一个java程序时,会启动一个java虚拟机进程.同一个jvm的所有线程.所有变量都处于同一个进程里,都使用该jvm进程的内存区. jvm进程终 ...

  8. Java的类加载器

    一.类加载器的概念 类加载器(class loader)用来加载 Java 类到 Java 虚拟机中.一般来说,Java 虚拟机使用 Java 类的方式如下:Java 源程序(.java 文件)在经过 ...

  9. Java的类加载器种类(双亲委派)

    Java类加载器采用双亲委派模型: 1.启动类加载器:这个类加载器负责放在<JAVA_HOME>\lib目录中的,或者被-Xbootclasspath参数所指定的路径中的,并且是虚拟机识别 ...

随机推荐

  1. JS 中如何判断 undefined 和 null

    JS 中如何判断 undefined JavaScript 中有两个特殊数据类型:undefined 和 null,下节介绍了 null 的判断,下面谈谈 undefined 的判断. 以下是不正确的 ...

  2. 转 FileStream Read File

    FileStream Read File [C#] This example shows how to safely read file using FileStream in C#. To be s ...

  3. [转]注释驱动的 Spring cache 缓存介绍

    原文:http://www.ibm.com/developerworks/cn/opensource/os-cn-spring-cache/ 概述 Spring 3.1 引入了激动人心的基于注释(an ...

  4. Alpha阶段事后分析报告

    每个团队编写一个事后分析报告,对于团队在Alpha阶段的工作做一个总结. 请在2016年11月24日上课之前根据下述博客中的模板总结前一阶段的工作,发表在团队博客上,并在课上的事后分析会上进行汇报,并 ...

  5. 开源 XFControls , 用于 Xamarin.Forms 的自定义控件集

    从此以后不会在博客园上发表任何言论,观注我的同志们,洗洗睡吧. ---------------------- 博文移至: http://www.jianshu.com/p/3ed1a3f10955

  6. 学堂在线 UWP 首版

    好久没有写博客了,主要是最近在写一个小小的App.<( ̄︶ ̄)> 不知道看各位有木有爱看慕课的,作为一名资深的大三学渣的我有看慕课的习惯.一直在看学堂在线的慕课,感觉质量确实还可以,但是遗 ...

  7. vNext之旅(1):从概念和基础开始

    ASP.NET vNext or .NET vNext? vNext在曝光以来绝大多数以ASP.NET vNext这样的的字眼出现,为什么这边会提及.NET vNext?原因是我认为ASP.NET只是 ...

  8. Linux vmstat命令实战详解

    vmstat命令是最常见的Linux/Unix监控工具,可以展现给定时间间隔的服务器的状态值,包括服务器的CPU使用率,内存使用,虚拟内存交换情况,IO读写情况.这个命令是我查看Linux/Unix最 ...

  9. ArcGIS支持MongoDB数据源

    ArcGIS支持MongoDB数据源 自从NoSQL推出之后,MongoDB就作为比较杰出的代表受到广大用户的推崇,当然,与之而来的大数据的讨论也非常激烈,GIS数据源向来都是以海量来计算,所以,GI ...

  10. 新玩具---Amazon Kindle PaperWhite 2

    自从将闲置了一段时间的K3 Keyboard 3G送人后,就一直用Nexus7平板上装Kindle程序来读书,用着也挺好,没有出现很多人说的费眼问题,说来也奇怪上学毕业之后,一直从事编程相关的工作有七 ...