代理模式之动态代理

动态代理是指动态的生成代理类 真实对象和接口是已经存在了的

1、动态代理和静态代理的角色是一样的;
2、动态代理的代理类是动态生成的;
3、分为两类,一类是基于接口的动态代理,一个是基于类的动态代理

a) 基于接口的动态代理-----jdk动态代理
b) 基于类的动态代理----cglib

4、jdk动态代理

关键的两个类:   proxy类 和 invocationHandler接口

invocationHandler 是 代理实例的 调用处理程序 实现的接口,
有一个方法是 invoke (object proxy , Method method object [ ] args );

该方法 在代理实例上 处理方法调用 并返回结果;
proxy -- 被代理的对象;
method -- 对应的是代理实例调用的接口方法的Method实例;(被代理的方法)
args -- 方法调用所需要的参数数组

proxy 提供用于创建动态代理类和实例的静态方法
//伪代码
其中有个重要的方法 newProxyInstance(类加载器, 代理类要实现的接口,invocationHandler)

代码示例:

//出租的抽象接口  
public interface Rent {  
    void Rent();  
}  
//被代理的角色  
public class Host implements Rent{  
    @Override  
    public void Rent() {  
        System.out.println("房东要租房");  
    }  
}  
//代理类  
public class ProxyInvocationHandler implements InvocationHandler {  
        private Object target;  
        public void setTarget(Object target) {  
            this.target = target;  
        }  
    @Override  
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {  
        seeHouse();  
        Object result = method.invoke(target, args);  
        fare();  
        return result;    
    }  
    public Object getProxy(){  
        /** 
         * 利用jdk的Proxy这个类来为我们动态的生成代理类 
         * this.getClass().getClassLoader(),  通过这个本身这个类的类加载器来加载新的代理类 
         * target.getClass().getInterfaces(), 得到抽象类的接口 
         * this  需要的是InvocationHandler  而ProxyInvocationHandler 这个类实现了InvocationHandler 
         */  
        return Proxy.newProxyInstance(this.getClass().getClassLoader(),   
                target.getClass().getInterfaces(), this);     

    public void seeHouse(){  
        System.out.println("中介带房东看房");  
    }  
    public void fare(){  
        System.out.println("中介收取一波中介费");  
    }  
}
//测试类  
public class Test {  
    public static void main(String[] args) {  
            Host host = new Host();  
            ProxyInvocationHandler ih = new ProxyInvocationHandler();   
            ih.setTarget(host);  
            Rent r = (Rent)ih.getProxy();  
            r.Rent();  
    }  
}
中介带房东看房
增加一名新用户
执行了:add方法
中介收取一波中介费

项目中稍作改进:
我们增加UserService和UserServiceImpl实现类;
在ProxyInvocationHandler中 我们模拟增加打印日志的功能,增加log();

public class ProxyInvocationHandler implements InvocationHandler {  
        private Object target;  
        public void setTarget(Object target) {  
            this.target = target;  
        }  
    @Override  
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {  
        seeHouse();  
        Object result = method.invoke(target, args);  
        log(method.getName());  
        fare();  
        return result;  
    }  
    public Object getProxy(){  
        /** 
         * 利用jdk的Proxy这个类来为我们动态的生成代理类 
         * this.getClass().getClassLoader(),  通过这个本身这个类的类加载器来加载新的代理类 
         * this.getClass().getInterfaces(),   得到实现 
         * this  需要的是InvocationHandler  而proxy这个类实现了InvocationHandler 
         */  
        return Proxy.newProxyInstance(this.getClass().getClassLoader(),   
                target.getClass().getInterfaces(), this);  
          
    }  
    public void log(String  MethodName){  
        System.out.println("执行了:"+MethodName+"方法");  
    }  
    public void seeHouse(){  
        System.out.println("中介带房东看房");  
    }  
    public void fare(){  
        System.out.println("中介收取一波中介费");  
    }  
}
public class Test {  
    public static void main(String[] args) {  
            UserService user = new UserServiceImpl();  
            ProxyInvocationHandler ih = new ProxyInvocationHandler();   
            ih.setTarget(user);  
            UserService r = (UserService)ih.getProxy();  
            r.add();  
    }  
} 
中介带房东看房  
增加一名新用户  
执行了:add方法  
中介收取一波中介费  

可以看到 打印了 执行了:add方法 这就是我们常用的SpringAOP来做日志管理的原理;

你可能感兴趣的:(代理模式之动态代理)