黑马程序猿——Java中的类载入器
------- android培训、java培训、期待与您交流!
--------
类载入器
Java虚拟机中能够安装多个类载入器,系统默认三个主要类载入器,每一个类负责载入特定位置的类:
BootStrap ExtClassLoader AppClassLoader
类载入器有什么作用?
当程序须要的某个类,那么须要通过类载入器把类的二进制载入到内存中,类载入器也是Java类
类载入器也是java类,由于其它是java类的类载入器本身也要被类载入器载入。显然必须有第一个类载入器不是java类。这正是BootStrap
Java虚拟机中的全部类载入器採用具有父子关系的树型结构进行组织。在实例化每一个类载入器对象时,须要为其指定一个父级类载入器对象或者默认採用系统类载入器为其父级类载入器。
类载入器的托付机制:
当java虚拟机要载入一个类时,究竟派出哪个类载入器去载入呢?
(1) 首先当前线程的类载入器去载入线程中的第一个类
(2) 假设类A中引用了类B,java虚拟机将使用载入类A装载器来载入类B
(3) 还能够直接调用ClassLoader.loadClass()方法来指定某个类载入器去载入某个类
watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvSW5zaXN0T3du/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" alt="">
面试题:能不能自己写个类java.lang.System?
通常不能够写。由于类载入器採用托付机制,这样保证了父类优先,也就是说总是使用父类能找到的类,这样总是使用java系统提供的System
编写自己的类载入器
知识解说:
(1) 自己定义的类载入器必须继承ClassLoader
(2) 覆写loadClass()方法下的findClass方法,由于要保留loadClass中的机制
(3) defineClass方法
编程步骤:
(1) 编写一个对文件内容进行简单加密的程序
(2) 编写一个自己的类载入器,可实现对加密过的类进行装载和解密
(3) 编写一个程序调用类载入器载入类。在源程序中不能用该类名定义引用变量,由于编译器无法识别这个类。
程序中能够除了使用ClassLoader。load方法之外,还能够使用设置线程的上下文类载入器或者系统类载入器,然后再使用Class.forName
实验步骤:
(1) 对不带包名的class文件进行加密。加密结果存放到另外一个文件夹,比如,java MyClassLoader MyTest.class F:\itcast
(2) 执行载入类的程序,结果可以被正常载入,但打印出来的类装载器名称为AppClassLoader:java MyClassLoader MyTest F:\itcast
(3) 用加密后的类文件替换CLASSPATH环境下的类文件,再运行上一步操作就出问题了,错误说明是AppClassLoader类载入器装载失败
(4) 删除CLASSPATH环境下的类文件,再运行上一步操作就没问题了
编写自己的类载入器
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);
fis.close();
fos.close();
}
/**
* 加密方法,同一时候也是解密方法
* @param ips
* @param ops
* @throws Exception
*/
private static void cypher(InputStream ips ,OutputStream ops) throws Exception{
int b = -1;
while((b=ips.read())!=-1){
ops.write(b ^ 0xff);//假设是1就变成0,假设是0就变成1
}
}
然后在新建一个类,通过上面的方法将新建的类的字节码进行加密:
public class ClassLoaderAttachment extends Date {
public String toString(){
return "hello,itcast";
}
}
那么这就须要使用我们自己的类载入器来进行解密了.
public class MyClassLoader extends ClassLoader{
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);
fis.close();
fos.close();
}
private static void cypher(InputStream ips ,OutputStream ops) throws Exception{
int b = -1;
while((b=ips.read())!=-1){
ops.write(b ^ 0xff);//假设是1就变成0,假设是0就变成1
}
}
private String classDir;
@Override
protected Class<?> findClass(String name) throws ClassNotFoundException {
String classFileName = classDir + "\\" + name.substring(name.lastIndexOf('.')+1) + ".class";
try {
FileInputStream fis = new FileInputStream(classFileName);
ByteArrayOutputStream bos = new ByteArrayOutputStream();
cypher(fis,bos);
fis.close();
System.out.println("aaa");
byte[] bytes = bos.toByteArray();
return defineClass(bytes, 0, bytes.length);
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
public MyClassLoader(){
}
public MyClassLoader(String classDir){
this.classDir = classDir;
}
}
測试执行代码:
Class clazz = new MyClassLoader("myClass").loadClass("ClassLoaderAttachment");
//此处不能在使用ClassLoaderAttachment由于一旦用了之后,
//系统的类载入器就会去载入,导致失败,所以该类就继承了Date类了.
Date date = (Date)clazz.newInstance();
System.out.println(date);
执行结果:
------- android培训、java培训、期待与您交流!
--------
黑马程序猿——Java中的类载入器的更多相关文章
- 黑马程序猿————Java基础日常笔记---反射与正則表達式
------Java培训.Android培训.iOS培训..Net培训.期待与您交流! ------- 黑马程序猿----Java基础日常笔记---反射与正則表達式 1.1反射 反射的理解和作用: 首 ...
- 黑马程序猿——JAVA高新技术——反射
----------android培训.java培训.java学习型技术博客.期待与您交流!------------ 一.对于反射的概念 对于JAVA反射机制是在执行状态中,对于随意一个类.都可以知道 ...
- 黑马程序猿——JAVA基础——集合
----------android培训.java培训.java学习型技术博客.期待与您交流.------------ 一.关于java中的集合类 首先看一下,大致的框架流程图 ...
- 黑马程序猿——java基金会--jdk、变量
学习内容: 1.Java发展历史 2.jdk和jre的差别,功能. 3.jdk和jre的下载和安装 4.配置环境.path和classpath 5.helloworld程序 6.进制之间的转换 7.凝 ...
- 黑马程序猿——————java基础
一.软件开发 软件是什么? 软件是简单的来说,计算机数据和指令的集合,数据(比方年龄,性别).指令及时告诉计算机怎样对他进行处理.计算机但是没有人那么聪明啊! 二.图形化界面(GUI),主要特点就是. ...
- 黑马程序猿——JAVA面向对象的特性:封装,继承,多态
- ----------android培训.java培训.java学习型技术博客.期待与您交流!------------ ...
- 黑马程序猿——JAVA基础——IO流
----------android培训.java培训.java学习型技术博客.期待与您交流!------------ 一. 一.IO流的三种分类方式 1.按流的方向分为:输入流和输出流 2.按流的数 ...
- 黑马程序猿-----Java之你不得不知道的排序
------<a href="http://www.itheima.com" target="blank">Java培训.Android培训.iOS ...
- 黑马程序猿 ---------- Java网络技术之 ---正則表達式 (Day06)
---------------------- ASP.Net+Unity开发..Net培训.期待与您交流. ---------------------- 正則表達式 正則表達式:基本知识 1 字符, ...
随机推荐
- 什么是xsi:type ???
http://www.w3.org/2001/XMLSchema-instance http://help.eclipse.org/indigo/index.jsp?topic=%2Forg.ecli ...
- c++ 函数的函数声明
c++ 函数的函数声明 只要在被调用函数的首部的末尾加一个分号,就成为对该函数的函数声明.函数声明的位置应当在函数调用之前. #include <iostream> using names ...
- Java多线程初学者指南(9):为什么要进行数据同步
Java中的变量分为两类:局部变量和类变量.局部变量是指在方法内定义的变量,如在run方法中定义的变量.对于这些变量来说,并不存在线程之间共享的问题.因此,它们不需要进行数据同步.类变量是在类中定义的 ...
- shell 基础 $(cd `dirname $0`;pwd)
$ cd `dirname $0` 和PWD%/* shell变量的一些特殊用法 在命令行状态下单纯执行 $ cd `dirname $0` 是毫无意义的.因为他返回当前路径的"." ...
- 防御XSS攻击的七条原则
本文将会着重介绍防御XSS攻击的一些原则,需要读者对于XSS有所了解,至少知道XSS漏洞的基本原理,如果您对此不是特别清楚,请参考这两篇文章:<Stored and Reflected XSS ...
- 【网络流24题】No.18 分配问题 (二分图最佳匹配 费用流|KM)
[题意] 有 n 件工作要分配给 n 个人做.第 i 个人做第 j 件工作产生的效益为 cij . 试设计一个将n 件工作分配给 n 个人做的分配方案, 使产生的总效益最大. 输入文件示例input. ...
- URPF技术白皮书
URPF技术白皮书 摘 要:本文介绍了URPF的应用背景,URPF主要用于防止基于源地址欺骗的网络攻击行为,例如基于源地址欺骗的DoS攻击和DDoS攻击:随后介绍了URPF的技术原理以及URPF ...
- HDU-1701 Binary Tree Traversals
http://acm.hdu.edu.cn/showproblem.php?pid=1710 已知先序和中序遍历,求后序遍历二叉树. 思路:先递归建树的过程,后后序遍历. Binary Tree Tr ...
- 离散数学A
自反性:(都自指)所有的点自己指向自己[<a,a><b,b>]:反自反性:(都不自指)所有的点都绝不自己指向自己:对称性:但凡指,定互指[<a,b>,<b,a ...
- 在Raspberry配置优化安装LNMP环境总结
在Raspberry配置优化安装LNMP环境总结 apt-get update apt-get install nginx apt-get install php5-fpm php5-cli php5 ...