Java静态代理和动态代理

    今天介绍一下代理设计模式,在业务场景中使用代理模式的好处有很多,包括什么权限校验,事务管理等等,具体有什么好处大家自动百度吧,我这里只解释代理模式的设计原理。首先这个设计模式出来的时候先是静态代理模式,只有理解了静态代理模式,才能理解JDK提供的动态代理是怎么回事,首先介绍静态代理,如果有错误的地方请大家不要拍砖,请大家指正。  

    一,静态代理 

    第一种情况:假如我去租房,没有经过中介,直接找到了房屋的主人。这里抽象一下就是我直接访问了真实对象(房屋的主人),然后我把房屋给租了。

    第二种情况:假如我去租房,需要经过中介,找到一家租房的中介公司,然后把房屋租了。这里抽象一下就是我访问了真实对象的代理对象(中介),这里的代理对象(中介)需要持有真实对象(房屋主人)才能把房屋租给我(因为中介手里没有房子,他肯定要有房屋真实主人的房子才行)。结构图如下:

                 

    大家一定要明白的是,代理对象(中介)一定要持有真实对象(房屋主人)才行。

    下边是静态代理的代码实现:

    首先代理对象和真实对象都需要实现同一个接口或继承抽象类,这个接口或者抽象类其实就是说我这里是租房屋的,不是租汽车的或别的东西,接口中定义一个quote(报价)方法。   

public interface Subject {
    /***
     * 这个接口定义一个报价方法
     */
    public String quote();

}

    然后定义一个真实对象类,就是代表真实的房屋的主人,实现Subject接口,同时实现报价方法:

public class RealSubject implements Subject {
    @Override
    public String quote() {
        return "房屋出租2000元一个月";
    }
}

    定义一个代理类,这个代理类一定持有真实对象的引用

public class ProxySubject implements Subject{
    @Override
    public String quote() {
        //代理对象持有真实对象
        Subject subject = new RealSubject();
        //返回真实对象的报价多少钱
        String quote = subject.quote();
        //代理对象增加真实对象的报价
        return quote.replace("2000", "3000");
    }
}

  编写我们的测试类,租房去找中介,然后查看房屋的报价是多少钱

public static void main(String[] args) {
        /**
         * 代理对象和真实对象都要实现同一个接口
         */
        //直接访问真实对象
        Subject subject = new RealSubject();
        System.out.println(subject.quote());
        
        //访问代理对象
        subject = new ProxySubject();
        System.out.println(subject.quote());
    }

    执行之后会发现中介把房屋的租金增加了,这就起到了代理的作用。这就是一个静态代理的小例子,静态代理主要注意两点:1,真实类和代理类都实现一个相同的接口;2,代理类持有真实类的对象的引用。按照租房的情况说明就是:中介和房租主人都必须是卖房子的,不能你去中介租房,结果给你返回一个汽车的报价;中介必须有房屋主人的房子才能租给你,如果中介手中没有房子,只是一个空壳子,也没有办法把房子租给你。

    静态代理的一个不好的地方是如果接口里增加了一个方法,那么真实类和代理类都必须实现相应的方法,增加了代码的复杂程度。还有就是如果代理类没有实现接口怎么办(解决办法自行百度吧 ^_^)

    二,动态代理

    JDK提供了一个java.lang.reflect.Proxy对象,可以用来对真实类实现动态代理,在运行时通过反射增强真实类。我们这里给Subject接口增加一个方法  getHouseAddress(得到房子的地址),然后真实类实现这个方法,返回"房屋的地址在地球";代码如下:

    

public class RealSubjectHome implements Subject{

    @Override
    public String quote() {
        return "房屋出租2000元";
    }

    @Override
    public String getHouseAddress() {
        return "房屋的地址在地球";
    }
}

     这里我们看一下代理类的实现(主要的解释都在注释中):

public final class DynamicProxy implements InvocationHandler{
    
    //代理对象是要持有真实对象
    private Object RealObject;
    
    public Object createSubject(Object RealObject){
        /**
         * 参数含义:
         *             classloader 是代理的类加载器
         *             interface   就是代理类实现的接口,因为代理类需要和真实对象实现相同的接口,所以返回的就是真实类的接口
         *             InvocationHandler 调用真实类的处理函数,代理类需要对真实对象做一些处理
         *             返回值   返回一个真实类的代理对象
         * 
         */
        this.RealObject = RealObject;
        return Proxy.newProxyInstance(this.getClass().getClassLoader(), 
                                      this.RealObject.getClass().getInterfaces(), //这里是真实对象实现的接口,代理对象也要实现相同的接口
                                      this);
    }
    /**
     * 参数:
     *     proxy  就是代理类的对象
     *  method 就是需要调用的实现接口中的方法
     *  args   方法中的参数
     *  返回值    返回代理类调用实现接口的方法的返回值
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args)
            throws Throwable {
        //调用真实对象中的方法,获取返回值
        if(method.getName().equals("quote")){
            String returnValue = (String) method.invoke(this.RealObject, args);
            returnValue = returnValue.replace("2000", "3000");
            return returnValue;
        }
        //其他不需要代理类修改的方法
        return method.invoke(this.RealObject, args);
    }
}

      其中Proxy.newProxyInstance方法返回一个代理对象,其中的第三个参数是实现InvocationHandler的接口,当调用代理对象的方法的时候,会调用InvocationHandler接口中的invoke方法用来对真实对象的方法增强,我们在invoke方法中只修改了获得房屋报价的内容。对获得房屋地址方法没有进行更改,直接调用真实对象的方法就行了。解释一下就是中介会把房屋的出租价格提高,但是房屋的地址不能修改吧,所以动态代理可以很好的控制需要增强的方法。

      看一下我们的测试类:

public static void main(String[] args) {
        
        //找代理出租房屋
        RealSubjectHome subjecthome = new RealSubjectHome();
        DynamicProxy proxy = new DynamicProxy();
        Subject subject = (Subject) proxy.createSubject(subjecthome);
        //得到房屋的报价信息
        System.out.println(subject.quote());
        //得到房屋的地址
        System.out.println(subject.getHouseAddress());
    }

    使用动态代理编写代理类的时候,其实并不难,只要注意两点就行了:1,代理类持有真实类的引用 2,代理类和真实类实现同一个接口。然后要多proxy类的newProxyInstance的参数含义理解,对InvocationHandler中的invoke方法的参数理解,就可以了。

    动态代理的思想在框架中使用的很多,其实使用jdk提供的动态代理的实现并不是很多,都是使用三方的实现,例如CGLIB,直接对类的字节码文件进行增强。静态代理和动态代理的实现就先这样,如果叙述中有错误,请指明。

 

posted @ 2013-12-18 16:09  悠悠小竹子  阅读(1154)  评论(1编辑  收藏  举报