各类加载器虽然以父子相称,但是没有继承关系
(视频教程推荐:java课程)
点入ClassLoader的源码查看样例:
*登录后复制* class NetworkClassLoader extends ClassLoader { * String host; * int port; * * public Class findClass(String name) { * byte[] b = loadClassData(name); * return defineClass(name, b, 0, b.length); * } * * private byte[] loadClassData(String name) { * // load the class data from the connection * . . . * } * } *
只需继承CalssLoader,重写 findClass方法即可:
立即学习“Java免费学习笔记(深入)”;
package reflect; import java.io.File; import java.io.FileInputStream; import java.io.IOException; import java.io.InputStream; import java.util.ArrayList; import java.util.List; public class MyClassLoader extends ClassLoader{ // 用于保存加载的文件的大小 private int classSize; // 新增的类加载路径 private static List登录后复制paths = new ArrayList(); static { paths.add("D:\Users\jeff.chan\Desktop\"); } // 默认会将app类加载器做为该类加载器的父加载器 public MyClassLoader() { } public MyClassLoader(ClassLoader parent) { super(parent); } @Override protected Class> findClass(String name) throws ClassNotFoundException { byte[] b = loadBinaryData(name); // 打印执行标记 System.out.println("use custom classloader..."); return defineClass(name, b,0, classSize); } public byte[] loadBinaryData(String name) throws ClassNotFoundException{ String pathName = name.replaceAll("\.", "/"); File file = null; for (String path : paths) { file = new File(path+pathName+".class"); if(file.exists() && file.isFile()){ break; } } if(file==null || !file.isFile()){ throw new ClassNotFoundException(); } // 读取文件的二进制信息 try (InputStream in = new FileInputStream(file)){ // 一般这个大小够了,这里是为了方便读取 // 文件的内容,所以一次性读入,并且将大小返回 // 用于defineClass byte[] b = new byte[1024 * 1024 * 10]; classSize = in.read(b); return b; }catch (IOException ioException){ ; } return null; } }
这里加入了一个MyClassLoader自定义类加载器,扫描的路径为:
D:\Users\jeff.chan\Desktop\登录后复制
那么新加入的类加载器就和app类加载器,扩展类加载器,启动类加载器一起起作用。
测试用例:
package reflect; public class Person { private String name; private String age; public Person() { } public Person(String name, String age) { this.name = name; this.age = age; } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getAge() { return age; } public void setAge(String age) { this.age = age; } @Override public String toString() { return "Person{" + "name='" + name + ''' + ", age='" + age + ''' + '}'; } }登录后复制
客户端:
package reflect; public class Client { public static void main(String[] args) throws Exception { MyClassLoader classLoader = new MyClassLoader(); Class> aClass = classLoader.loadClass("reflect.Person"); System.out.println(aClass); System.out.println(aClass.getClassLoader()); } }登录后复制
时直接运行代码:那么会输出
class reflect.Person
sun.misc.Launcher$AppClassLoader@18b4aac2
发现自己的类加载器根本不起作用。其实是因为双亲委托机制:
1、MyClassLoader 查看是否已经加载 -》 2、App类加载查看是否已经加载 -》3、 扩展类加载器查看是否已经加载
-》 4、启动类加载器查看是否已经加载,没有加载就尝试加载 -》 5、扩展类尝试加载 -》 6、app类加载尝试加载 -》 7、MyClassLoader尝试加载 -》 8、抛出找不到类的异常
上述的流程中,第6步会找到对应的类,所以就直接返回了,根本没有给MyClassLoader机会加载,所以才没有调用自定义的类加载器MyClassLoader。
如果想要加载的话,那么就到编译生成的Person.class移动到桌面reflect/Person.class下,执行结果:
use custom classloader...
class reflect.Person
reflect.MyClassLoader@135fbaa4
结果发现,自定义的类加载起作用了,其实就是上述的流程中的第7步,找到了相应的类的二进制文件,加载了进来。
相关推荐:java入门
以上就是java实现自定义类加载器的详细内容,更多请关注慧达安全导航其它相关文章!
免责 声明
1、本网站名称:慧达安全导航
2、本站永久网址:https//www.huida178.com/
3、本站所有资源来源于网友投稿和高价购买,所有资源仅对编程人员及源代码爱好者开放下载做参考和研究及学习,本站不提供任何技术服务!
4、本站所有资源的属示图片和信息不代表本站的立场!本站只是储蓄平台及搬运
5、下载者禁止在服务器和虚拟机下进行搭建运营,本站所有资源不支持联网运行!只允许调试,参考和研究!!!!
6、未经原版权作者许可禁止用于任何商业环境,任何人不得擅作它用,下载者不得用于违反国家法律,否则发生的一切法律后果自行承担!
7、为尊重作者版权,请在下载24小时内删除!请购买原版授权作品,支持你喜欢的作者,谢谢!
8.若资源侵犯了您的合法权益,请持 您的版权证书和相关原作品信息来信通知我们!QQ:1247526623我们会及时删除,给您带来的不便,我们深表歉意!
9、如下载链接失效、广告或者压缩包问题请联系站长处理
10、如果你也有好源码或者教程,可以发布到网站,分享有金币奖励和额外收入!
11、本站资源售价只是赞助,收取费用仅维持本站的日常运营所需
12、因源码具有可复制性,一经赞助,不得以任何形式退款。
13、本文内容由网友自发贡献和站长收集,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系1247526623@qq.com
转载请注明出处: 慧达安全导航 » java实现自定义类加载器
发表评论 取消回复