`

如何实现自己的classloader

    博客分类:
  • java
阅读更多

如何实现自己的classloader

如何实现自己的classloader来加载类      刚开始学习加载的时候,接触到的是HeloWorld程序,当时不知道为什么在public static void main(String [] args) 方法里写了System.out.println("Hello World!") 就可以在控制台打出“Hello World!”来,确实的说,是什么东西隐蔽在后面执行了我们写的这段代码,通过后来的学习,知道了所有的class都是通过classloader来加载的。java规范这么说,Java的ClassLoader就是用来动态装载class的,ClassLoader对一个class只会装载一次,JVM使用的ClassLoader一共有4种:
          启动类装载器,标准扩展类装载器,类路径装载器和网络类装载器。
这4种ClassLoader的优先级依次从高到低,使用所谓的“双亲委派模型”。确切地说,如果一个网络类装载器被请求装载一个java.lang.Integer,它会首先把请求发送给上一级的类路径装载器,如果返回已装载,则网络类装载器将不会装载这个java.lang.Integer,如果上一级的类路径装载器返回未装载,它才会装载java.lang.Integer。

        再说说Package权限。Java语言规定,在同一个包中的class,如果没有修饰符,默认为Package权限,包内的class都可以访问。但是这还不够准确。确切的说,只有由同一个ClassLoader装载的class才具有以上的Package权限。比如启动类装载器装载了java.lang.String,类路径装载器装载了我们自己写的java.lang.Test,它们不能互相访问对方具有Package权限的方法。这样就阻止了恶意代码访问核心类的Package权限方法。     
        现在来通过扩展ClassLoader类实现一个自己的类装载器,每个Class对象都有一个引用指向装载他的ClassLoader,可以通过public ClassLoader getClassLoader()方法得到它。为了创建自己的类装载器我们应该扩展ClassLoader类,这是一个抽象类。假设要从本地文件系统使用我们实现的类装载器装载一个类,创建一个FileClassLoader extends ClassLoader,需要覆盖ClassLoader中的findClass(String name)方法,这个方法通过类的名字而得到一个Class对象。

package test.International;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
public class FileClassLoader extends ClassLoader{
    
public static final String drive = "d:/";
    
public static final String fileType = ".class";
    
public FileClassLoader() {
         super();
     }

    
public FileClassLoader(ClassLoader arg0) {
         super(arg0);
     }

    
public Class findClass(String name) {
        
byte[] data = loadClassData(name);
        
return defineClass(name, data, 0, data.length);
     }

    
public byte[] loadClassData(String name) {
         FileInputStream fis
= null;
        
byte[] data = null;
        
try {
             fis
= new FileInputStream(new File(drive + name + fileType));
             ByteArrayOutputStream baos
= new ByteArrayOutputStream();
            
int ch = 0;
            
while ((ch = fis.read()) != -1) {
                 baos.write(ch);
             }

             data
= baos.toByteArray();
         }
catch (IOException e) {
             e.printStackTrace();
         }

        
return data;
     }

    
public static void main(String[] args) throws Exception {
         FileClassLoader loader
= new FileClassLoader();
         Class objClass
= loader.loadClass("HelloWorld", true);
         Object obj
= objClass.newInstance();
         System.
out.println(objClass.getName());
         System.
out.println(objClass.getClassLoader());
     }

}

现在把HelloWorld.java放到默认包下,然后编译,把HelloWorld.class放到D:\盘根目录下,运行FileClassLoader类,控制台输出:

HelloWorld
sun.misc.Launcher$AppClassLoader@82ba41

可见,我们的确加载上了HelloWorld.class

分享到:
评论
1 楼 sotrip 2012-10-05  
从上面结果来看,HelloWorld.class是由虚拟机的内置装载器加载的(sun.misc.Launcher$AppClassLoader@82ba41),

不是我们的FileClassLoader加载的

要把HelloWorld.class放在ClassPath之外,这样虚拟机的内置装载器就装载不了了,就可以通过我们的FileClassLoader来加载

相关推荐

    ClassLoader运行机制 自己写的

    ClassLoader运行机制 自己写的ClassLoader运行机制 自己写的ClassLoader运行机制 自己写的ClassLoader运行机制 自己写的ClassLoader运行机制 自己写的

    classloader的简单实现

    一个开源的Cplusplus类加载器,基于它实现了一个简单的例子,见我写的classloader的文章。

    ClassLoader 案例

    自定义ClassLoader,控制台输入调试。 运行期间 重新载入指定目录的class文件。可实现对于类的功能函数更新。 用到java 反射,@interface 等技术

    webshpere classloader 原理

    详细说明了websphere classloader 的架构与实现

    Java实现热加载完整代码;Java动态加载class;Java覆盖已加载的class;Java自定义classloader

    让Java支持热加载是个不错的想法。如何做到的呢? 1. 定义好接口和实现类 2. 让代理类通过反射的方式调用实现类,对外暴露的是代理类。 3. 自定义URLClassLoader。检查实现类.class文件的...Java自定义classloader;

    my_classloader_demo-master.7z Native层实现热修复

    my_classloader_demo-master.7z Native层实现热修复

    Android 使用classloader原理进行热更新

    使用Android的classloader加载器实现热更新,通过反射机制获取到源码的Elements数组替换classes.dex实现更新,只能重启软件进行更新,无法实现实时更新。

    CustomClassLoader源码

    利用JAVA中的ClassLoader实现热部署

    Java ClassLoader学习总结

    主要内容包括 Java类加载机制及加载流程,以及如何定义自己的类加载器,如何实现类的热替换。

    对Java的ClassLoader的几点认识

     2、ClassLoader的实现类URLClassLoader完成工作的方法是根据传入的class目录或jar文件的URL读取二进制文件,并转化为Class;  3、ClassLoader有个成员变量parent,使用ClassLoader获取Class时,默认会先用它的...

    service_classloader

    工程名为service_classloader,功能全部使用corejava实现,使用maven管理依赖,没有使用任何框架(junit除外)。 使用的技术主要包括:classlaoder,加密解密,线程,ThreadLocal使用,反射,IO操作,jdbc, 序列化,...

    ClassLoaderDemo.7z

    以一个实际的小例子,演示关于使用ClassLoader实现自定义多组件的加载Demo,可以帮助理解类加载器的双亲委托及破坏。

    ClassLoader实例

    比如某些类我们需要动态的加载进java虚拟机的内存区域。 要实现这个功能我们就需要了解java虚拟机的几个类加载器。

    reload-java:在Java 8中尝试使用ClassLoader

    重新载入Java 支持类重载的ClassLoader的简单实现运行示例构建项目开始示例 gradle buildjava -cp build/classes/test/:build/libs/reload-java-1.0.jar reload.ReloadExample1该应用程序将等待输入,输入r + 已加载...

    安卓原生热更新 classloader

    安卓原生热更新,具体实现方法,运行APK后,修改里面的Toast文字,打包APK解压出classe.dex文件,把文件导入手机,点击修复后重启软件就能完成热更新

    classloader类加载器_基于java类的加载方式详解

    下面小编就为大家带来一篇classloader类加载器_基于java类的加载方式详解。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧

    restx-classloader-0.33.1.zip

    TubeSock.zip,用java a实现的websocket客户端用java实现的websocket客户端库

    java面试题宝典

    面试题集共分为以下十部分: 一、Core Java: 1 — 95 题1 — 24 页 基础及语法: 1 — 61 题1 — 13 页 异常: 62 — 69 题13 — 15 页 集合: 70 — 80 题15 — 18 页 线程: 81 — 90 题18 — 21 页 ...

    class-loader:[不推荐] ClassLoader组件提供了一些工具,可以自动加载类并缓存其位置以提高性能

    Universal ClassLoader能够自动加载实现PSR-0标准或PEAR命名约定的类。 首先,注册自动装带器: require_once __DIR__.'/src/Symfony/Component/ClassLoader/UniversalClassLoader.php'; use Symfony\Component\...

    详解Android类加载ClassLoader

    Java ClassLoader BootstrapClassLoader 引导类加载器 ,用来加载Java的核心库。通过底层代码来实现的,基本上只要parent为null,那就表示引导类加载器。 比如:charsets.jar、deploy.jar、javaws.jar、jce.jar、jfr...

Global site tag (gtag.js) - Google Analytics