第一步:  给你一个编译好的class文件以及它的包名,创建一个对象出来。

1)class文件源代码

  1. package com.wsc.classloader;
  2. public class Tool{
  3. public void print() {
  4. }
  5. }

2)使用javac Tool.java 编译成class文件

3)将Tool.class文件读取到内存中,生成byte[]数组

  1. /**
  2. * 加载class文件
  3. *
  4. * @param clazzPath
  5. *            class绝对文件路径
  6. * @return 字节数组
  7. * @throws IOException
  8. */
  9. private byte[] loadClassFile(String clazzPath) throws IOException {
  10. FileInputStream fis = new FileInputStream(clazzPath);
  11. BufferedInputStream bis = new BufferedInputStream(fis);
  12. ByteArrayOutputStream baos = new ByteArrayOutputStream();
  13. byte[] buffer = new byte[1024 * 256];
  14. int ch = 0;
  15. while ((ch = bis.read(buffer, 0, buffer.length)) != -1) {
  16. baos.write(buffer, 0, ch);
  17. }
  18. return baos.toByteArray();
  19. }

4)自定义ClassLoader,使用ClassLoader中的defineClass方法:protected final Class<?> defineClass(String name, byte[] b, int off, int len)。参数分别是类名称,class文件对应的字节数组,起始位置和终止位置。

  1. @Override
  2. protected Class<?> loadClass(String name, boolean resolve)
  3. throws ClassNotFoundException {
  4. Class<?> c = findLoadedClass(name);
  5. if (c == null) {
  6. c = defineClass(name, data, 0, data.length);
  7. }
  8. return c;
  9. }

整体代码是:

  1. package com.wsc.classloader;
  2. import java.io.BufferedInputStream;
  3. import java.io.ByteArrayOutputStream;
  4. import java.io.FileInputStream;
  5. import java.io.IOException;
  6. public class ClassLoaderOne extends ClassLoader {
  7. public static void main(String[] args) throws Exception {
  8. ClassLoaderOne loader = new ClassLoaderOne(
  9. "E:\\JAVA\\JAVAFX\\ClassLoader\\libs\\Tool.class");
  10. Class<?> clazz = loader.loadClass("com.wsc.classloader.Tool");
  11. Object o = clazz.newInstance();
  12. System.out.println(o.getClass().getClassLoader());
  13. }
  14. private byte[] data;
  15. public ClassLoaderOne(String clazzPath) throws IOException {
  16. data = loadClassFile(clazzPath);
  17. }
  18. /**
  19. * 加载class文件
  20. *
  21. * @param clazzPath
  22. *            class绝对文件路径
  23. * @return 字节数组
  24. * @throws IOException
  25. */
  26. private byte[] loadClassFile(String clazzPath) throws IOException {
  27. FileInputStream fis = new FileInputStream(clazzPath);
  28. BufferedInputStream bis = new BufferedInputStream(fis);
  29. ByteArrayOutputStream baos = new ByteArrayOutputStream();
  30. byte[] buffer = new byte[1024 * 256];
  31. int ch = 0;
  32. while ((ch = bis.read(buffer, 0, buffer.length)) != -1) {
  33. baos.write(buffer, 0, ch);
  34. }
  35. return baos.toByteArray();
  36. }
  37. @Override
  38. protected Class<?> loadClass(String name, boolean resolve)
  39. throws ClassNotFoundException {
  40. Class<?> c = findLoadedClass(name);
  41. if (c == null) {
  42. c = defineClass(name, data, 0, data.length);
  43. }
  44. return c;
  45. }
  46. }

感觉是这样的,跑一下:

  1. Exception in thread "main" java.lang.SecurityException: Prohibited package name: java.lang
  2. at java.lang.ClassLoader.preDefineClass(Unknown Source)
  3. at java.lang.ClassLoader.defineClass(Unknown Source)
  4. at java.lang.ClassLoader.defineClass(Unknown Source)
  5. at com.wsc.classloader.ClassLoaderOne.loadClass(ClassLoaderOne.java:52)
  6. at java.lang.ClassLoader.loadClass(Unknown Source)
  7. at java.lang.ClassLoader.defineClass1(Native Method)
  8. at java.lang.ClassLoader.defineClass(Unknown Source)
  9. at java.lang.ClassLoader.defineClass(Unknown Source)
  10. at com.wsc.classloader.ClassLoaderOne.loadClass(ClassLoaderOne.java:52)
  11. at java.lang.ClassLoader.loadClass(Unknown Source)
  12. at com.wsc.classloader.ClassLoaderOne.main(ClassLoaderOne.java:14)

意思是:禁止加载名为java.lang的包。

原因是:虽然Tool类中没有使用任何引入java.lang下类,但是它的父类Object是在java.lang下的,classloader加载Tool类时会把它所有的关系网都加载出来才行,父类Object肯定是要加载的。

这样就简单了!无非多写一个If else.使用父加载器(类加载器都有一个父类加载器)加载即可。

  1. @Override
  2. protected Class<?> loadClass(String name, boolean resolve)
  3. throws ClassNotFoundException {
  4. Class<?> c = findLoadedClass(name);
  5. if (name.equals("java.lang.Object")) {
  6. ClassLoader parent = getParent();
  7. c = parent.loadClass(name);
  8. }
  9. if (c == null) {
  10. c = defineClass(name, data, 0, data.length);
  11. }
  12. return c;
  13. }

跑一下结果是:

  1. com.wsc.classloader.ClassLoaderOne@ca470

第二步:新的问题

  1. Method[] methods = clazz.getMethods();
  2. for (int i = 0; i < methods.length; i++) {
  3. String name = methods[i].getName();
  4. System.out.println(name);
  5. Class<?>[] params = methods[i].getParameterTypes();
  6. for (int j = 0; j < params.length; j++) {
  7. System.out.println(params[j].toString());
  8. }
  9. }

这个时候还是会报刚才的错误,因为Method类也在java.lang包下,只能在增加一个If else.

显然,代码应该这样写

  1. @Override
  2. protected Class<?> loadClass(String name, boolean resolve)
  3. throws ClassNotFoundException {
  4. Class<?> c = findLoadedClass(name);
  5. if (c == null) {
  6. // 如果父加载器不为null,使用父类加载器加载(比如Object,HashMap等核心类)
  7. if (getParent() != null) {
  8. try {
  9. c = getParent().loadClass(name);
  10. } catch (Exception e) {
  11. // 父类可能没加载,则抛异常
  12. }
  13. }
  14. // 如果父类加载器没有加载,再使用自定义加载器加载
  15. if (c == null) {
  16. c = defineClass(name, data, 0, data.length);
  17. }
  18. }
  19. return c;
  20. }

打印结果:

  1. toString
  2. print
  3. class java.lang.String
  4. getClass
  5. hashCode
  6. equals
  7. class java.lang.Object
  8. notify
  9. notifyAll
  10. wait
  11. long
  12. int
  13. wait
  14. wait
  15. long
  16. com.wsc.classloader.ClassLoaderOne@fcfa52

第三步:如果本地可以通过.class文件创建,远程当然也已同一个道理(如果需要加密,在本地多一个解密即可)。如果class文件是远程调用的话,本地一般使用接口或者反射两种方法调用。首选是接口,反射一是效率,而是要清楚所有的方法名称、参数名称过于麻烦。

由于远程加载class文件到本地,如果出错很难定位出错位置。幸好,classloader使用规则默认是根据URLClassLoader来使用的,会先根据检查本地是否有该类,所以可以直接将源码放在本地即可调试,当然发布的时候一定要删除。

如图:

    通过这个基本的入门程序可以了解ClassLoader的基本流程。

1

用.class文件创建对象的更多相关文章

  1. 大三上 —— IOS五天实训

    第二天: 注册使用xib:1.首先为xib文件创建对象--let nib = UINib(nibName: "xib文件名", bundle: nil).2.具体的控件注册该xib ...

  2. xib与nib的区别

    xib和nib都是Interface Builder的图形界面设计文档,nib这个名字来自于NeXTSTEP系统,在NeXTSTEP被Apple收购之前,一直使用nib作为Interface Buil ...

  3. 十七、EnterpriseFrameWork框架核心类库之Web控制器

    回<[开源]EnterpriseFrameWork框架系列文章索引> EFW框架源代码下载:http://pan.baidu.com/s/1qWJjo3U EFW框架中的WebContro ...

  4. VBS基础篇 - wscript 对象

    一.wscript对象 描述:提供对 Windows 脚本宿主对象模型根对象的访问.详述:WScript 对象是 Windows 脚本宿主对象模型层次结构的根对象.它可在任何脚本文件中使用,不需要特定 ...

  5. 【转】基于RMAN实现坏块介质恢复(blockrecover)

    本文转自:乐沙弥的世界 对于物理损坏的数据块,我们可以通过RMAN块介质恢复(BLOCK MEDIA RECOVERY)功能来完成受损块的恢复,而不需要恢复整个数据库或所有文件来修复这些少量受损的数据 ...

  6. java反射知识

    java反射机制是在运行状态中,对于任意一个类(class文件),都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取的信息以及动态调用对象的方法的功能称 ...

  7. 我用Cocos2d-x模拟《Love Live!学院偶像祭》的Live场景(二)

    上一章分析了Live场景中各个元素的作用,这一章开始来分析最关键的部分——打击物件的实现. 上一章放出的视频很模糊,先来看一个清晰版的复习一下:http://www.bilibili.com/vide ...

  8. 【Unity 3D 游戏开发】Unity3D 入门 - 工作区域介绍 与 入门示例

    一. 工作区域详解 1. Scence视图 (场景设计面板) scence视图简介 : 展示创建的游戏对象, 可以对所有的游戏对象进行 移动, 操作 和 放置; -- 示例 : 创建一个球体, 控制摄 ...

  9. OC语言大总结(上)

    根据OC学习的知识点,总结了一下,希望能帮到大家! 作者:韩俊强  未经允许,请勿转载! 关注博主:http://weibo.com/hanjunqiang 第一节类于对象   类与对象http:// ...

随机推荐

  1. [CSS]float&clear浮动

    CSS float 属性 浮动的框可以向左或向右移动,直到它的外边缘碰到包含框或另一个浮动框的边框为止. 由于浮动框不在文档的普通流中,所以文档的普通流中的块框表现得就像浮动框不存在一样.  可取的值 ...

  2. python中的字典应用实例

    字典中的键使用时必须满足一下两个条件: 1.每个键只能对应一个项,也就是说,一键对应多个值时不允许的(列表.元组和其他字典的容器对象除外).当有键发生冲突时(即字典键重复赋值),取最后的赋值. > ...

  3. WPF 核心体系结构

    WPF 体系结构 主题提供 Windows Presentation Foundation (WPF) 类层次结构,涵盖了 WPF 的大部分主要子系统,并描述它们是如何交互的. System.Obje ...

  4. struts中如何将前台的值能在action中获取到

    如何获取值----三种方式(属性驱动,对象驱动,模型驱动)  A:属性驱动 必须生成get,set方法  B:对象驱动 给对象也必须生成get,set方法  c模型驱动 模型驱动需要action去实现 ...

  5. FTP原理和cent OS vsFTPd架设

    1.ftp为明码传输 2.客户端和服务端采用两条链路来分别进行命令和数据的传输.数据传输的模式分为主动链接和被动链接. 3.客户端在需要数据的时候,会告知服务器端采取主动或者被动的方式来链接. 4.如 ...

  6. Java OAuth开发包资料

    原文地址:http://www.oschina.net/project/tag/307/oauth?lang=19&sort=time

  7. Scanner 与 Readable 的read()方法

    Readable接口中的read()方法实现了将字符串读入charBuffer中,但是只有在需要输出的时候才会调用. Scanner是文本扫描器类,利用Scanner扫描并输出charBuffer中的 ...

  8. [置顶] linux内核启动2-setup_arch中的内存初始化(目前分析高端内存)

    上一篇微博留下了这几个函数,现在我们来分析它们         sanity_check_meminfo();         arm_memblock_init(&meminfo, mdes ...

  9. [BZOJ2173]整数的lqp拆分

    [题目描述] lqp在为出题而烦恼,他完全没有头绪,好烦啊… 他首先想到了整数拆分.整数拆分是个很有趣的问题.给你一个正整数N,对于N的一个整数拆分就是满足任意m>0,a1 ,a2 ,a3…am ...

  10. Welcome Back to C++ (Modern C++)

    http://msdn.microsoft.com/en-us/library/hh279654.aspx 看看你是古代还是现代Cpper重点强调 Stack-based scope instead ...