Java中的静态代理、通用动态代理类以及原理剖析(三)

2014-11-23 21:50:19 · 作者: · 浏览: 64
ode 函数 : surfTheInternet 函数 : gotoFacebook 实现的接口 : com.thingking.in.java.proxy.Network class java.lang.reflect.Proxy 可以看到network所属的类是com.thingking.in.java.proxy.$Proxy1,其中.$Proxy之前是我的demo的报名,$Proxy1是network的类名。然后这个类实现的函数有equals, toString, hashCode, surfTheInternet, gotoFacebook这几个函数,前三个都是Object类的方法,后两个是Network接口的方法。实现的接口是Network,其父类是jdk中的java.lang.reflect.Proxy。输出的结果正好验证了我们上面所说的。 下面是Proxy.newProxyInstance方法的简单模拟实现 ( 实际情况当然没有那么简单 ) :

[java] view plaincopy 在CODE上查看代码片 派生到我的代码片
  1. import java.io.File;
  2. import java.io.FileWriter;
  3. import java.lang.reflect.Constructor;
  4. import java.lang.reflect.Method;
  5. import java.net.URL;
  6. import java.net.URLClassLoader;
  7. import javax.tools.JavaCompiler;
  8. import javax.tools.StandardJavaFileManager;
  9. import javax.tools.ToolProvider;
  10. import javax.tools.JavaCompiler.CompilationTask;
  11. public class Proxy {
  12. public static Object newProxyInstance(ClassLoader loader,Class infce, InvocationHandler h) throws Exception { //JDK6 Complier API, CGLib, ASM
  13. String methodStr = "";
  14. String rt = "\r\n";
  15. //利用反射,获得infce接口中方法,本例中就是获得Moveable接口的方法move
  16. Method[] methods = infce.getMethods();
  17. //拼接infce中所有方法字符串,用来重写infce中的方法,本例中拼出来就是重写的move方法
  18. for(Method m : methods) {
  19. methodStr += "@Override" + rt +
  20. "public void " + m.getName() + "() {" + rt +
  21. " try {" + rt +
  22. " Method md = " + infce.getName() + ".class.getMethod(\"" + m.getName() + "\");" + rt +
  23. /*方法最核心的代码,h是构造$Proxy1时传入的handler,本例中就是LogHandler对象,new Object[] { null}是move方法需要的参数,本例不需要,故为空。这一步将会使我们编写的处理类逻辑LogHandler的invoke方法得到调用。从而达到我们最初要在move方法前加日志逻辑的的目的,下面要做的就是把我们拼好的字符串生成类并load到内存就可以了这样就实现了动态生成代理类并加自己想加的逻辑*/
  24. " h.invoke(this, md,new Object[] { null});" + rt +
  25. " }catch(Exception e) {e.printStackTrace();}" + rt +
  26. "}";
  27. }
  28. String src =
  29. "package com.bjsxt.proxy;" + rt +
  30. "import java.lang.reflect.Method;" + rt +
  31. //这里动态实现infce接口,本例中就是Moveable,构造方法中让Proxy持有处理类Handler的引用
  32. "public class $Proxy1 implements " + infce.getName() + "{" + rt +
  33. " public $Proxy1(InvocationHandler h) {" + rt +
  34. " this.h = h;" + rt +
  35. " }" + rt +
  36. " com.bjsxt.proxy.InvocationHandler h;" + rt +
  37. //这里是需要重写Moveable中的方法,见上面该字符串的拼接过程
  38. methodStr +
  39. "}";
  40. String fileName =
  41. "d:/src/com/bjsxt/proxy/$Proxy1.java";
  42. File f = new File(fileName);
  43. FileWriter fw = new FileWriter(f);
  44. fw.write(src);
  45. fw.flush();
  46. fw.close();
  47. //compile编译上面拼好的字符串
  48. JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
  49. StandardJavaFileManager fileMgr = compiler.getStandardFileManager(null, null, null);
  50. Iterable units = fileMgr.getJavaFileObjects(fileName);
  51. CompilationTask t = compiler.getTask(null, fileMgr, null, null, null, units);
  52. t.call();
  53. fileMgr.close();
  54. //load into memory and create an instance加载进内存并创建对象
  55. URL[] urls = new URL[] {new URL("file:/" + "d:/src/")};
  56. URLClassLoader ul = new URLClassLoader(urls);
  57. Class c = ul.loadClass("com.bjsxt.proxy.$Proxy1");
  58. System.out.println(c);
  59. Constructor ctr = c.getConstructor(InvocationHandler.class);
  60. Object m = ctr.newInstance(h);
  61. //利用反射,这里就返回一个实现了infce接口也就是本例中Moveable接口的代理对像$Proxy1
  62. return m;
  63. }
  64. } 需要注意的是这里的Proxy类和JDK中java.lang.reflect.Proxy并不相同。只是他们的原理大致相同,实现的细节并不同。JDK中生成$Proxy1并不是拼字符串,而是直接生成二进制码。