• 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实现机制
  • 相关阅读:
    初学angular
    C#二维数组及其本质(转)
    蓝桥杯之递归算法
    同时安装Office2016和Visio2016
    蓝桥杯之暴力破解、枚举
    NAT(地址解析协议)
    云中继
    ACL(访问控制列表)
    虚拟链路(virtual-link)
    单臂路由
  • 原文地址:https://www.cnblogs.com/huligong1234/p/3353561.html
Copyright © 2020-2023  润新知