关于ClassLoad的技术(动态装载)

本文介绍使用类库加载器ClassLoader在运行时刻调入class的方法,包括写接口类、实现接口文件、将编译后的文件内容放入byte[]、转换成Class并初始化等步骤,还提到可对文件加解密,但Java无类库卸载器,运行期间无法更新class,最后指出部署到Jboss存在类找不到的问题。
部署运行你感兴趣的模型镜像

/*在运行时刻从文件中调入Class (ClassLoader 的使用)
本文介绍如何使用类库加载器ClassLoader 实现在运行时刻调入class

方法1
  使用接口类,新调用的class是对它的具体实现
1) 写一个接口类 newface.class
2) 写接口文件实现 testfacea.class 更名为 testfacea.file 或其它文件名全可以
3) 主程序中调入文件到byte[]中,可以在本地文件调用,也可用网络无论如何只要能将
   编译后的文件内容的类代码放到 byte[]当中就可以
4) 转换成一个Class并初始化
5) 实现拉口
实际上就是对一个接口类用调入的文件实现,当然可以用不同的文件进行不同的实现
也可以对一个文件进行加解密操作,
要注意的是对一个要调入的文件,一定要是一个已经存在的接口类的实现
这个有点EJB中的调用的中远程接口要在本地,而调入的文件就是EJBobject了
这种方法的的好处是要调入的class中的方法是可以说是已知的,相对来讲这种方法
简明易用


其它:
但是Java并不提供一个类似于类库卸载器(ClassUnloader) 的功能,能够把已经装载的模块从内存
里面清除掉。所以如果一个class已经调入,就放在了内存当中,并在程序信息表内记录本class已经
调入,如果再一次使用就从内存当中调入,如果实际文件更新也只能是程序下次运行时间再调入,而不
能在运行期间更新一个class,如果强行再一次调入就会
Linkage Error: duplicate class definition (重复定义class)

*/
package com;
//使用的主程序
public class testnewface {
    public static void main(String[] args) throws java.lang.Exception
     {

//  共用初使化参数,开始
    Class testc;
    Object testo;
    cloader cl=new cloader();
//  共用初使化参数,结束

//方法 1 的例子代码,newface是本地接口类,newface.class本地已经存在 开始
    testc=cl.load("testfacea.class","testfacea");
    testo=testc.newInstance();

     System.out.println("Instance has over!");
    ((newface)testo).out("方法1 第(1)种使用方法");
     System.out.println("outsize="+((newface)testo).outsize("1111","aaaa"));
    }
}
//接口
package com;

public interface newface {

       public void out(String xx);
       public int outsize(String x1,String x2);

 }
//实现接口的类--编译后的.class文件放到D://
package com;
public class testfacea implements newface{

    public void out(String xx)  {
    System.out.println(xx+" for testfacea ");
    }
    public int outsize(String x1,String x2)  {
      return x1.length()+x2.length();
    }

}

//自己的ClassLoad类Loader
package com;

/*
要想自己完成从一个 byte[] 转换到一个Class 必须要 extends ClassLoader
因为ClassLoader中的方法defineClass是 protected  要使用只有 extends ClassLoader
*/
//import com.newface;
class cloader extends ClassLoader  {
     static int maxsize=10000;
      public Class load(String namefile,String classname) throws java.lang.Exception
       {
           Class result=null;
          try {

           //进行判断这个class是否已经调入,已经有就直接返回,不然就调入
         Class ctmp=this.findLoadedClass(classname);
           System.out.println(ctmp.getName()+" is load");
           return ctmp;
           }
           catch (Exception e)  {
           System.out.println("memory is null");
             }
        System.out.println("start..");
        java.io.FileInputStream in=new java.io.FileInputStream("d://"+namefile);
        byte[] classbyte=new byte[maxsize];
//实际应用时完全可以对一个文件进行加解密处理,只要保证使用defineClass时classbyte中
//已经解密后的内容就可以
        int readsize;
        readsize=in.read(classbyte);
        System.out.println("读文件长:"+readsize);
        in.close();
        System.out.println("the classname:"+classname);
        return defineClass("com."+classname,classbyte,0,readsize);
              }
}

********************************************************************
注意:编译后的.class文件的包名,需要同我们的文件的是同一个包下的。
*********************************************************************
部署到Jboss 上还存在问题,即我们的类找不到!!
*********************************************************************
注:以上是我们根据网上某网友ClassLoad的示例进行的修改。大家共同交流。
*********************************************************************

您可能感兴趣的与本文相关的镜像

GPT-oss:20b

GPT-oss:20b

图文对话
Gpt-oss

GPT OSS 是OpenAI 推出的重量级开放模型,面向强推理、智能体任务以及多样化开发场景

由于没有具体的引用内容,下面从通用角度介绍`ClassLoad.java`相关信息。 ### 代码解读 `ClassLoad`通常与Java的类加载机制相关。在Java中,类加载器(`ClassLoader`)负责将类的字节码文件加载到Java虚拟机(JVM)中。以下是一个简单的`ClassLoad`示例代码及解读: ```java import java.net.URL; import java.net.URLClassLoader; public class ClassLoad { public static void main(String[] args) { try { // 创建一个URLClassLoader URL[] urls = new URL[]{new URL("file:/path/to/your/classes/")}; URLClassLoader classLoader = new URLClassLoader(urls); // 加载指定类 Class<?> clazz = classLoader.loadClass("com.example.MyClass"); // 创建该类的实例 Object instance = clazz.getDeclaredConstructor().newInstance(); System.out.println("Class loaded and instance created: " + instance); } catch (Exception e) { e.printStackTrace(); } } } ``` - `URLClassLoader`:用于从指定的URL路径加载类。这里`urls`数组指定了类文件所在的路径。 - `loadClass`方法:用于加载指定全限定名的类。 - `newInstance`方法:用于创建该类的实例。 ### 使用方法 1. **自定义类加载器**:当需要从非默认路径加载类,或者实现自定义的类加载逻辑时,可以继承`ClassLoader`类,并重写`findClass`方法。 ```java import java.io.ByteArrayOutputStream; import java.io.File; import java.io.FileInputStream; import java.io.IOException; public class CustomClassLoader extends ClassLoader { @Override protected Class<?> findClass(String name) throws ClassNotFoundException { try { String filePath = name.replace('.', File.separatorChar) + ".class"; File classFile = new File(filePath); FileInputStream fis = new FileInputStream(classFile); ByteArrayOutputStream bos = new ByteArrayOutputStream(); byte[] buffer = new byte[1024]; int bytesRead; while ((bytesRead = fis.read(buffer)) != -1) { bos.write(buffer, 0, bytesRead); } byte[] classData = bos.toByteArray(); fis.close(); bos.close(); return defineClass(name, classData, 0, classData.length); } catch (IOException e) { throw new ClassNotFoundException(name, e); } } } ``` 2. **使用自定义类加载器**: ```java public class Main { public static void main(String[] args) { CustomClassLoader customClassLoader = new CustomClassLoader(); try { Class<?> clazz = customClassLoader.loadClass("com.example.MyClass"); Object instance = clazz.getDeclaredConstructor().newInstance(); System.out.println("Class loaded and instance created: " + instance); } catch (Exception e) { e.printStackTrace(); } } } ``` ### 错误解决 - **ClassNotFoundException**:通常是由于类路径错误或者类名拼写错误导致的。检查类的全限定名和类文件所在的路径是否正确。 - **NoClassDefFoundError**:这可能是因为类的依赖项缺失,或者类加载器无法找到所需的类。确保所有依赖的类都在类路径中。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值