Java代理学习笔记
代理模式
代理模式是常用的java设计模式,他的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类,以及事后处理消息等。代理类与委托类之间通常会存在关联关系,一个代理类的对象与一个委托类的对象关联,代理类的对象本身并不真正实现服务,而是通过调用委托类的对象的相关方法,来提供特定的服务。
按照代理的创建时期,代理类可以分为两种。
静态代理:由程序员创建或特定工具自动生成源代码,再对其编译。在程序运行前,代理类的.class文件就已经存在了。
动态代理:在程序运行时,运用反射机制动态创建而成。
AOP(面向切面编程),就是针对代理的一种应用。
下面以一个简单示例开始:
public interface Hello { void say(String name); } public class HelloImpl implements Hello { @Override public void say(String name) { System.out.println("Hello! "+name); } } ---------------main------------------------ Hello hello = new HelloImpl(); hello.say("Jack"); ---------------console------------------------ Hello! Jack
现在有个需求,在say()方法执行前后分别执行相关逻辑。下面分别通过静态代理和动态代理来实现。 1.静态代理(GoF代理模式)
public class HelloProxy implements Hello { private Hello hello; public HelloProxy(){ hello = new HelloImpl(); } @Override public void say(String name) { before(); hello.say(name); after(); } private void before(){ System.out.println("Before"); } private void after(){ System.out.println("After"); } } ---------------main------------------------ Hello hp = new HelloProxy(); hp.say("Tom"); ---------------console------------------------ Before Hello! Tom After
2.动态代理(JDK动态代理,CGLIB动态代理) (1).JDK动态代理
import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; public class DynamicProxy implements InvocationHandler { private Object target; public DynamicProxy(Object target){ this.target = target; } @SuppressWarnings("unchecked") public <T> T getProxy() { return (T) Proxy.newProxyInstance( target.getClass().getClassLoader(), target.getClass().getInterfaces(), this ); } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { before(); Object result = method.invoke(target, args); after(); return result; } private void before(){ System.out.println("Before"); } private void after(){ System.out.println("After"); } } ----------------main----------------------- DynamicProxy dp = new DynamicProxy(new HelloImpl()); Hello helloProxy = dp.getProxy(); helloProxy.say("Grace"); -----------------console---------------------- Before Hello! Grace After
JDK的动态代理机制只能代理实现了接口的类,而不能实现接口的类就不能实现JDK的动态代理,cglib是针对类来实现代理的,他的原理是对指定的目标类生成一个子类,并覆盖其中方法实现增强,但因为采用的是继承,所以不能对final修饰的类进行代理。 (2).CGLib实现动态代理 需要引入两个jar: cglib-2.2.2.jar, asm-3.3.jar ASM:一个轻量级的字节码生成及转换器。 CGLIB:一个功能强大的动态代理代码工具,可以根据指定的类动态生成一个子类,并提供了方法拦截的相关机制,并且在大量的流行开源框架(如 Hibernate、Spring 等)中得到使用。
import java.lang.reflect.Method; import net.sf.cglib.proxy.Enhancer; import net.sf.cglib.proxy.MethodInterceptor; import net.sf.cglib.proxy.MethodProxy; public class CGLibProxy implements MethodInterceptor { private static CGLibProxy instance = new CGLibProxy(); private CGLibProxy() { } public static CGLibProxy getInstance() { return instance; } @SuppressWarnings("unchecked") public <T> T getProxy(Class<T> cls) { return (T) Enhancer.create(cls, this); } @Override public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable { before(); Object result = proxy.invokeSuper(obj, args); after(); return result; } private void before(){ System.out.println("Before"); } private void after(){ System.out.println("After"); } } ----------------main----------------------- HelloImpl helloImpl = CGLibProxy.getInstance().getProxy(HelloImpl.class); helloImpl.say("Victor"); ---------------console------------------------ Before Hello! Victor After
参考: 1.http://my.oschina.net/huangyong/blog/159788 Proxy那点事儿 2.http://my.oschina.net/huangyong/blog/161338 AOP那点事儿 3.http://jinnianshilongnian.iteye.com/blog/1474325 我对AOP的理解 4.http://kiral.iteye.com/blog/1198982 AOP实现机制