Java ClassLoader加载机制理解 实际例子
针对 Java ClassLoader加载机制理解, 做了个如何自定制简单的ClassLoader,并成功加载指定的类。
不废话,直接上代码。
package com.chq.study.cl; import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.channels.Channels;
import java.nio.channels.FileChannel;
import java.nio.channels.WritableByteChannel; /**
* @desc 自定义ClassLoader,只能加载.class结尾的,用来测试java的classLoader机制
*/
public class ChqClassLoader extends ClassLoader {
private String fileName; public ChqClassLoader(String fileName) {
this.fileName = fileName;
} protected Class<?> findClass(String className) throws ClassNotFoundException {
Class<?> clazz = this.findLoadedClass(className);
if (null == clazz) {
try {
String classFile = getClassFile(className);
System.out.println("findClass " + classFile);
FileInputStream fis = new FileInputStream(classFile);
FileChannel fileC = fis.getChannel();
ByteArrayOutputStream baos = new ByteArrayOutputStream();
WritableByteChannel outC = Channels.newChannel(baos);
ByteBuffer buffer = ByteBuffer.allocateDirect(1024);
while (true) {
int i = fileC.read(buffer);
if (i == 0 || i == -1) {
break;
}
buffer.flip();
outC.write(buffer);
buffer.clear();
}
fis.close();
byte[] bytes = baos.toByteArray(); clazz = defineClass(className, bytes, 0, bytes.length);
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
return clazz;
} private String getClassFile(String name) {
StringBuffer sb = new StringBuffer(fileName);
name = name.replace('.', File.separatorChar) + ".class";
sb.append(File.separator + name);
return sb.toString();
}
}
package com.chq.study.cl; /**
* @desc 自我介绍测试接口类,塑型用的
*/
public interface ITest {
public void self();
}
package com.chq.study.cl; /**
* @desc 未从接口类继承
*/
public class Test {
public void self() {
System.out.println("this is from Test instance " + this);
}
}
package com.chq.study.cl; /**
* @desc 自我介绍测试实现类
*/
public class TestImpl implements ITest { /* (non-Javadoc)
* @see com.chq.study.cl.ITest#self()
*/
@Override
public void self() {
System.out.println("this is from TestImpl instance " + this);
} }
package com.chq.study.cl; /**
* @author chenqing
* @datetime 2015年2月4日 下午4:54:12
* @desc 入口类, 调用自定义的ClassLoader,来加载类进行验证
*/
public class MainClassLoader { /**
* @param args
*/
public static void main(String[] args) {
ChqClassLoader cl = new ChqClassLoader("C:\\workspaces\\MyEclipse Professional 2014\\classloader\\bin");
try {
Class<?> clazz = cl.findClass("com.chq.study.cl.Test");
try {
// 此处执行会抛出异常,验证了classLoader的全盘负责机制
Test cc = (Test) clazz.newInstance();
cc.self();
System.out.println("belong class loader: " + cc.getClass().getClassLoader().toString());
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (ClassCastException e) {
e.printStackTrace();
} // 确保输出顺序
try {
Thread.sleep(100);
} catch (InterruptedException e1) {
e1.printStackTrace();
} clazz = cl.findClass("com.chq.study.cl.TestImpl");
try {
// 此处正常,通过塑性为基类来绕开全盘负责机制
ITest ic = (ITest)clazz.newInstance();
ic.self();
System.out.println("belong class loader: " + ic.getClass().getClassLoader().toString());
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (ClassCastException e) {
e.printStackTrace();
}
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
} }
最终输出的结果表明, Test & TestImpl 都找到了,但在实例化时,因为不同的classLoader加载的,导致前者失败,后者则通过塑型为基类而成功加载并实例化。
findClass C:\workspaces\MyEclipse Professional 2014\classloader\bin\com\chq\study\cl\Test.class
java.lang.ClassCastException: com.chq.study.cl.Test cannot be cast to com.chq.study.cl.Test
at com.chq.study.cl.MainClassLoader.main(MainClassLoader.java:22)
findClass C:\workspaces\MyEclipse Professional 2014\classloader\bin\com\chq\study\cl\TestImpl.class
this is from TestImpl instance com.chq.study.cl.TestImpl@173a10f
belong class loader: com.chq.study.cl.ChqClassLoader@14318bb
Java ClassLoader加载机制理解 实际例子的更多相关文章
- Java ClassLoader加载机制理解
今天看到了一篇介绍Java ClassLoader加载机器的文章, 才发觉一直来自己的肤浅, 好好地给补了一课, 不得不存档! 原文地址: http://www.blogjava.net/lhulcn ...
- Java ClassLoader加载机制
一.体系结构(自上向下) 1.Bootstrap ClassLoader(BootStrapClassLoader) --- 启动类加载器或者叫引导类加载器,加载jdk核心的APIs,这些APIs一般 ...
- java class加载机制及对象生成机制
java class加载机制及对象生成机制 当使用到某个类,但该类还未初始化,未加载到内存中时会经历类加载.链接.初始化三个步骤完成类的初始化.需要注意的是类的初始化和链接的顺序有可能是互换的. Cl ...
- java动态加载机制
假设有一个class,ClassLoader首先把它load到内存里的code segment(内存里存放代码段的),站在ClassLoader的角度,内存里的一个一个的class就是一个一个的对象, ...
- java内存加载机制
什么是java类加载? 类加载是指将.class类中的二进制数据存放到内存中,会在内存中的推中建立一个java.lang.String的引用对象来存放方法区的数据结构,而类中的数据会放到方法区中 类加 ...
- 深入理解ClassLoader(四)—类的父委托加载机制
上几次我们介绍到了JVM内部的几个类加载器,我们来重新画一下这个图,再来看一下他们之间的关系.
- 《深入理解java虚拟机》笔记(8)类的加载机制
一.类加载机制 类加载器将类的.class文件中的二进制数据读入到内存中,将其放在方法区内,然后在堆区创建一个java.lang.Class对象,用来封装类在方法区内的数据结构.类的加载的最终产品是位 ...
- jvm系列(一):java类的加载机制
java类的加载机制 1.什么是类的加载 类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个java.lang.Class对象,用来封装 ...
- Java 类的加载机制
1.什么是类的加载 类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个java.lang.Class对象,用来封装类在方法区内的数据结构 ...
随机推荐
- caffe使用finetume
训练时, solver.prototxt中使用的是train_val.prototxt ./build/tools/caffe/train -solver ./models/bvlc_referenc ...
- hive中行转换成列以及hive相关知识
Hive语句: Join应该把大表放到最后 左连接时,左表中出现的JOIN字段都保留,右表没有连接上的都为空.对于带WHERE条件的JOIN语句,例如: 1 SELECT a.val, b.val F ...
- linux下查看资源使用情况
//查看占用内存最多的前K的程序ps aux | sort -k4nr | head -K //查看占用CPU最多的前K的程序 ps aux | sort -k3nr | head -K
- hdu 5468(dfs序+容斥原理)
Puzzled Elena Time Limit: 5000/2500 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others)T ...
- DNS区域传送漏洞的安全案例
DNS区域传送(DNS zone transfer)指的是一台备用服务器使用来自主服务器的数据刷新自己的域(zone)数据库.这为运行中的DNS服务提供了一定的冗余度,其目的是为了防止主的域名服务 ...
- [实战]MVC5+EF6+MySql企业网盘实战(20)——Bootstrap Paginator
写在前面 考虑到数据量的问题,特引入bootstrap的分页插件.插件下载地址:https://github.com/lyonlai/bootstrap-paginator 系列文章 [EF]vs15 ...
- 实现手机端上下左右滑屏的jq原生代码和使用库·两种办法
先来一个原生的.我使用的是jq. 需要注意的地方就是被触发的元素最好不要是body,这个代码也可以修改,如果obj传进来的是body那么,$(this)必须是你的监听元素,不然会冒泡泡,整个项目就…… ...
- 使用matlab表示“段数不确定”的分段函数
示例函数: 分段函数f(x)的段数为数组a的长度减1,在表达f(x)时,不能直接使用a的长度5-1=4. 方法1: 先计算每个间隔点的函数值f(a2),f(a3),f(a4),再循环表示f(x). f ...
- LoadRunner中Vugen-Recording Options选项卡介绍:
LoadRunner中Vugen-Recording Options选项卡介绍:
- ubuntu sublime text 3 build 3083 license
经验证:sublime text 3 3083可用 ----- BEGIN LICENSE -----Andrew WeberSingle User LicenseEA7E-855605813A03D ...