------- android培训、java培训、期待与您交流! ----------
package com.qdzks;
import java.io.ByteArrayOutputStream;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.io.OutputStream;
//一下是通过自己的手抄笔记的整理,如果有错误,希望大家多多指正
/*
* 类加载器
·简要介绍什么事类加载器和类加载器的作用
·Java虚拟机中可以安装多个类加载器,系统默认三个主要的类加载器,每个类负责加载特定位置的类:
BootStrap,ExtClassLoader,AppClassLoader
·类加载器也是Java类,因为其是java类的类加载器本身也要被类加载器加载,显然必须有第一个类加载器不是java类,这正是BootStrap
·Java虚拟机中的所有类装载器采用具有父子关系的树形结构进行组织,在实例化每个类装载器对象时,需要为其指定一个父级类装载器对象或者默认采用系统类装载器为其父级类加载图
类加载器的委托机制
·当Java虚拟机要加载一个类时,到底派出哪儿类加载器去加载呢?
-首先当前线程的类加载器去加载线程中的第一个类。
-如果类A中引用了类B,Java虚拟机将使用加载类A的类装载器来加载类B。
-还可以直接调用ClassLoader.loadClass()方法来指定某个类加载器去加载某个类。
·每个类加载器加载类时,又先委托给其他上级类加载器。
-当所有祖宗类加载器没有加载到类,回到发起者类加载器,还加载不类,则抛ClassNotFoundException,不是再去找发起者类加载器的儿子,因为没有getChild方法,即使有,那有多个儿子,找哪一个呢?
-对着类加载器的层次结构图和委托加载原理,解释先前将ClassLoaderTest输出成jre/lib/ext目录下的lqqheima.jar后运行结果为:sun.misc.Launcher$ExtClassLoader
null
sun.misc.Launcher$ExtClassLoader
null
理解:
每个ClassLoader本身只能分别加载特定位置和目录的类,但它们可以委托其他的类装载器去加载类,这就是类加载器的委托模式。类装载器一级级委托到BootStrap类加载器,当BootStrap无法加载当前所要加载的类时,然后才一级级退回到子孙类装载器去进行真正的加载。当退回到最初的类装载器时,如果它自己也不能完成类的装载,那就应报告ClassNotFoundException异常。
有一道面试题:能不能自己写一个类加java.lang.System,为了不让我们写System类,类加载采用委托机制,这样可以保证爸爸们优先,也就是总是使用爸爸们能找到的类,这样总是使用java系统提供的System类。
编写自己的类加载器
* */
/*编写自己的类加载器
* 知识讲解:自定义类加载器必须继承classLoader
* .loadClass与findClass
* .defineClass方法
* 编程步骤
*-编写一个对文件内容进行简单加密的程序。
*-编写一个自己的类装载器,可实现对加密过的类进行装载和解密。
*-编写一个程序调用类加载器加载类,在源程序中不能用该类名定义引用变量,因为编译器无法识别这个类。程序中可以除类使用类 ClassLoader.load方法之外,还可以使用设置线程的上下文类加载器或者系统类加载器,然后再使用Class.forName。
* */
public class MyClassLoader extends ClassLoader{
/**
* @param args
*/
public static void main(String[] args) throws Exception{
// TODO Auto-generated method stub
//在主函数上传参 右击选择run as 选择run configrations 然后再Arguments 里写入你穿的参数并空格分隔开
//args[0]是要读取.class文件的目录
String srcPath = args[0];
//args[1]为要写入的目录
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);
//流用完之后记得要关
fis.close();
fos.close();
}
//对文件进行加密
private static void cypher(InputStream ips,OutputStream ops)throws Exception{
int b=0;
while((b=ips.read())!=-1){
/*小知识点讲解:
* 0xff 是十六进制的取8位 1111 1111 代表255
* b^0xff之后原先的代码就取反了 呵呵 这个知识点以前不太清楚
* */
ops.write(b^0xff);
}
}
private String classDir;
/*复写ClassLoader中findClass 方法 其实自己写类加载我感觉
* 就是查找类文件的目录,并将其解密之后 在内存中返回一个.class 文件
* */
@Override
protected Class<?> findClass(String name) throws ClassNotFoundException {
// TODO Auto-generated method stub
//.class文件的全路径
String classFileName = classDir + "\\"+name +".class";
try {
//对其进行读取
FileInputStream fis = new FileInputStream(classFileName);
//顺便回顾一下字节数组流对象 字节输出流的目的就是内存中的一个数组
/*所以不用指定目的 相同的还有CharArrayStream StringReader
* */
ByteArrayOutputStream bos = new ByteArrayOutputStream();
cypher(fis,bos);;
fis.close();
//这句话是看看有没有调用我的类加载器
System.out.println("aaa");
byte[] bytes = bos.toByteArray();
//讲数组转换成class文件 虽然这个方法过时了 但是还是感觉很厉害 呵呵 直接返回.class文件
return defineClass(bytes,0,bytes.length);
} catch (Exception e) {
// TODO: handle exception
}
//如果找不到,在调用父类的findClass方法
return super.findClass(name);
}
public MyClassLoader(){
}
public MyClassLoader(String classDir){
//这个是将存的.class文件的目录找出
this.classDir=classDir;
}
}