针对 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加载机制理解 实际例子的更多相关文章

  1. Java ClassLoader加载机制理解

    今天看到了一篇介绍Java ClassLoader加载机器的文章, 才发觉一直来自己的肤浅, 好好地给补了一课, 不得不存档! 原文地址: http://www.blogjava.net/lhulcn ...

  2. Java ClassLoader加载机制

    一.体系结构(自上向下) 1.Bootstrap ClassLoader(BootStrapClassLoader) --- 启动类加载器或者叫引导类加载器,加载jdk核心的APIs,这些APIs一般 ...

  3. java class加载机制及对象生成机制

    java class加载机制及对象生成机制 当使用到某个类,但该类还未初始化,未加载到内存中时会经历类加载.链接.初始化三个步骤完成类的初始化.需要注意的是类的初始化和链接的顺序有可能是互换的. Cl ...

  4. java动态加载机制

    假设有一个class,ClassLoader首先把它load到内存里的code segment(内存里存放代码段的),站在ClassLoader的角度,内存里的一个一个的class就是一个一个的对象, ...

  5. java内存加载机制

    什么是java类加载? 类加载是指将.class类中的二进制数据存放到内存中,会在内存中的推中建立一个java.lang.String的引用对象来存放方法区的数据结构,而类中的数据会放到方法区中 类加 ...

  6. 深入理解ClassLoader(四)—类的父委托加载机制

    上几次我们介绍到了JVM内部的几个类加载器,我们来重新画一下这个图,再来看一下他们之间的关系.

  7. 《深入理解java虚拟机》笔记(8)类的加载机制

    一.类加载机制 类加载器将类的.class文件中的二进制数据读入到内存中,将其放在方法区内,然后在堆区创建一个java.lang.Class对象,用来封装类在方法区内的数据结构.类的加载的最终产品是位 ...

  8. jvm系列(一):java类的加载机制

    java类的加载机制 1.什么是类的加载 类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个java.lang.Class对象,用来封装 ...

  9. Java 类的加载机制

    1.什么是类的加载 类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个java.lang.Class对象,用来封装类在方法区内的数据结构 ...

随机推荐

  1. word2vec参数

    架构:skip-gram(慢.对罕见字有利)vs CBOW(快) ·         训练算法:分层softmax(对罕见字有利)vs 负采样(对常见词和低纬向量有利) 负例采样准确率提高,速度会慢, ...

  2. AspNet Core 发布到Linux系统和发布IIS 注意项

    AspNet Core 发布到Linux系统和发布IIS 注意项 1.发布时需要注意的 2.Windows Server 2012 api-ms-win-crt-runtime-l1-1-0.dll ...

  3. MySQL 和 Oracle 主键自增长

    1.MySQL 1)建表 auto_increment:每插入一条数据,客户表(customers)的主键id就自动增1,如下所示 create table customers -- 创建客户表 ( ...

  4. 动手编写TCP服务器系列之一:日志文件

    前言 在几个月之前,笔者想自己实现一个性能比较良好的基于tcp的服务器.于是断断续续写了个把月,因为还需要找工,还有论文什么的.拖了这么久.现在开辟这样的一个博客,我想记录下自己的思路,也和大家分享自 ...

  5. [重磅]Deep Forest,非神经网络的深度模型,周志华老师最新之作,三十分钟理解!

    欢迎转载,转载请注明:本文出自Bin的专栏blog.csdn.net/xbinworld. 技术交流QQ群:433250724,欢迎对算法.技术感兴趣的同学加入. 深度学习最大的贡献,个人认为就是表征 ...

  6. mongodb卸载再重装

    标题就凸显了尴尬,是的,本地(ubuntu16.04)自带的mongodb太老了,想要装最新版的 卸载: sudo dpkg -P mongodb 然后下载新版的mongodb:  https://m ...

  7. sublime text3 如何在多行前面快速插入序号

    sublime text3 如何在多行前面快速插入序号 1.需要安装InsertNums插件 首选项 -> Package Control -> Install Package -> ...

  8. 获取 web 项目的绝对路径

    获取 web 项目的绝对路径 <% String path = request.getContextPath(); String basePath = request.getScheme()+& ...

  9. Mybatis 源码分析之一二级缓存

    一级缓存 其实关于 Mybatis 的一级缓存是比较抽象的,并没有什么特别的配置,都是在代码中体现出来的. 当调用 Configuration 的 newExecutor 方法来创建 executor ...

  10. 洛谷——P1907 设计道路

    P1907 设计道路 题目描述 Caesar远征高卢回来后,对你大加赞赏,他亲自来到Genoa视察. Genoa在你的建设下变得无比繁荣,由于财政收入的增加,你为城市修建了交通系统.古罗马的交通系统由 ...