code前行

  博客园  :: 首页  :: 新随笔  :: 联系 :: 订阅 订阅  :: 管理

CGLIB 使用相关资料:戳这里

回顾动态代理:参考资料http://blog.csdn.net/ykzhen2015/article/details/50312651   https://www.ibm.com/developerworks/cn/java/j-lo-proxy1/index.html

接口
public
interface HelloWorld { public String sayHi(); }
实现类

public class HelloWorldImpl implements HelloWorld {

@Override
public String sayHi() {
System.out.println("Hello World from Implement");
return "Hello World from Implement";
}

}

代理类
import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method;
public class HelloWorldProxy implements InvocationHandler { private Object originalObj;//目标类 public HelloWorldProxy(Object originalObj) { super(); this.originalObj = originalObj; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("方法调用前"); Object returnObject = method.invoke(originalObj, args); System.out.println("方法调用后"); return returnObject; } }
测试代码
import java.lang.reflect.InvocationHandler; import java.lang.reflect.Proxy;
public class ProxyTest { public static void main(String[] args) { HelloWorld helloworld = new HelloWorldImpl(); InvocationHandler proxyClass = new HelloWorldProxy(helloworld); HelloWorld helloworldProxy = (HelloWorld) Proxy.newProxyInstance(helloworld.getClass().getClassLoader(), helloworld.getClass().getInterfaces(), proxyClass); System.out.println(helloworldProxy.sayHi()); } }

动态生成的代理类本身的一些特点。

1、包:如果所代理的接口都是 public 的,那么它将被定义在顶层包(即包路径为空),如果所代理的接口中有非 public 的接口(因为接口不能被定义为 protect 或 private,所以除 public 之外就是默认的 package 访问级别),那么它将被定义在该接口所在包(假设代理了 com.ibm.developerworks 包中的某非 public 接口 A,那么新生成的代理类所在的包就是 com.ibm.developerworks),这样设计的目的是为了最大程度的保证动态代理类不会因为包管理的问题而无法被成功定义并访问;

2、类修饰符:该代理类具有 final 和 public 修饰符,意味着它可以被所有的类访问,但是不能被再度继承;

3、类名:格式是“$ProxyN”,其中 N 是一个逐一递增的阿拉伯数字,代表 Proxy 类第 N 次生成的动态代理类,值得注意的一点是,并不是每次调用 Proxy 的静态方法创建动态代理类都会使得 N 值增加,原因是如果对同一组接口(包括接口排列的顺序相同)试图重复创建动态代理类,它会很聪明地返回先前已经创建好的代理类的类对象,而不会再尝试去创建一个全新的代理类,这样可以节省不必要的代码重复生成,提高了代理类的创建效率。

4、类继承关系:该类的继承关系如图:

动态代理类的继承图

图 2. 动态代理类的继承图

可见,Proxy 类是它的父类,这个规则适用于所有由 Proxy 创建的动态代理类。而且该类还实现了其所代理的一组接口,这就是为什么它能够被安全地类型转换到其所代理的某接口的根本原因

 

被代理的一组接口注意:首先,要注意不能有重复的接口,以避免动态代理类代码生成时的编译错误。其次,这些接口对于类装载器必须可见,否则类装载器将无法链接它们,将会导致类定义失败。再次,需被代理的所有非 public 的接口必须在同一个包中,否则代理类生成也会失败。最后,接口的数目不能超过 65535,这是 JVM 设定的限制。

异常处理方面的特点:从调用处理器接口声明的方法中可以看到理论上它能够抛出任何类型的异常,因为所有的异常都继承于 Throwable 接口,但事实是否如此呢?答案是否定的,原因是我们必须遵守一个继承原则:即子类覆盖父类或实现父接口的方法时,抛出的异常必须在原方法支持的异常列表之内。所以虽然调用处理器理论上讲能够,但实际上往往受限制,除非父接口中的方法支持抛 Throwable 异常。那么如果在 invoke 方法中的确产生了接口方法声明中不支持的异常,那将如何呢?放心,Java 动态代理类已经为我们设计好了解决方法:它将会抛出 UndeclaredThrowableException 异常。这个异常是一个 RuntimeException 类型,所以不会引起编译错误。通过该异常的 getCause 方法,还可以获得原来那个不受支持的异常对象,以便于错误诊断

 

jdk动态代理和cglib动态代理的对比:两种方法同时存在,各有优劣。jdk动态代理是由java内部的反射机制来实现的,cglib动态代理底层则是借助asm来实现的。总的来说,反射机制在生成类的过程中比较高效,而asm在生成类之后的相关执行过程中比较高效(可以通过将asm生成的类进行缓存,这样解决asm生成类过程低效问题)。还有一点必须注意:jdk动态代理的应用前提,必须是目标类基于统一的接口。如果没有上述前提,jdk动态代理不能应用。

posted on 2018-03-10 22:52  code前行  阅读(163)  评论(0编辑  收藏  举报