【Java设计模式】代理模式

本文详细介绍了代理模式,包括其意图、优点和缺点。静态代理和动态代理的区别进行了阐述,其中静态代理需要手动创建代理类,而动态代理(如JDK和CGLIB实现)则在运行时生成。JDK动态代理基于接口实现,CGLIB代理则不需要接口,但无法重写final方法。代理模式在增加功能或控制访问等方面具有高扩展性,但也可能影响性能。
摘要由CSDN通过智能技术生成

在代理模式(Proxy Pattern)中,一个类代表另一个类的功能。这种类型的设计模式属于结构型模式。 我们创建具有现有对象的对象,以便向外界提供功能接口

意图:为其他对象提供一种代理以控制对这个对象的访问

主要解决:在直接访问对象时带来的问题,比如说:要访问的对象在远程的机器上。在面向对象系统中,有些对象由于某些原因(比如对象创建开销很大,或者某些操作需要安全控制,或者需要进程外的访问),直接访问会给使用者或者系统结构带来很多麻烦,我们可以在访问此对象时加上一个对此对象的访问层,即增加中间层

在这里插入图片描述
优点:高扩展性

缺点:客户端和真实主题之间增加了代理对象,因此有些类型的代理模式可能会造成请求的处理速度变慢

代理模式分为静态代理、动态代理

1. 静态代理

静态代理,代理类在编译时生成;即,需要编码代理类。

代理接口:

package man.kuke.proxy.demo;

public interface Action {
    void watchTV();
}

被代理类:

package man.kuke.proxy.demo;

public class Person implements Action{

    @Override
    public void watchTV() {
        System.out.println("you are watching 《the dark world》!");
    }

}

静态代理类:

package man.kuke.proxy.demo;


public class StaticProxy implements Action{
    private Action action;

    public StaticProxy(Action action) {
        this.action = action;
    }

    @Override
    public void watchTV() {
        actiionBefore();
        action.watchTV();
        actionAfter();
    }

    private void actiionBefore() {
        System.out.println("you are drink water");
    }

    private void actionAfter() {
        System.out.println("you are shitting");
    }
}


在这里插入图片描述

2.动态代理

代理类在程序运行时创建的代理方式被成为动态代理。

2.1JDK 代理
public class DynamicProxy {

    @SuppressWarnings("unchecked")
    public static <T> T getProxy(Object obj,Class<T> klass) {
        return (T) Proxy.newProxyInstance(klass.getClassLoader(),new Class[]{klass},new InvocationHandler(){

            @Override
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                actiionBefore();
                method.invoke(obj,args);
                actionAfter();
                return null;
            }
        });
    }

    private static void actiionBefore() {
        System.out.println("you are drink water");
    }

    private static void actionAfter() {
        System.out.println("you are shitting");
    }
}

在这里插入图片描述

2.1.1 JDK动态代理

生成代理对象的字节码方法:

 System.getProperties().put("sun.misc.ProxyGenerator.saveGeneratedFiles","true");
public final class $Proxy0 extends Proxy implements Action {
    private static Method m1;
    private static Method m2;
    private static Method m3;
    private static Method m0;

 static {
        try {
            m1 = Class.forName("java.lang.Object").getMethod("equals", Class.forName("java.lang.Object"));
            m2 = Class.forName("java.lang.Object").getMethod("toString");
            m3 = Class.forName("man.kuke.proxy.demo.Action").getMethod("watchTV");
            m0 = Class.forName("java.lang.Object").getMethod("hashCode");
        } catch (NoSuchMethodException var2) {
            throw new NoSuchMethodError(var2.getMessage());
        } catch (ClassNotFoundException var3) {
            throw new NoClassDefFoundError(var3.getMessage());
        }
    }
    
  public $Proxy0(InvocationHandler var1) throws  {
        super(var1);
    }
}

可以看到,代理对象通过反射获被代理对象所有的方法。

实现了接口方法:

 public final void watchTV() throws  {
        try {
            super.h.invoke(this, m3, (Object[])null);
        } catch (RuntimeException | Error var2) {
            throw var2;
        } catch (Throwable var3) {
            throw new UndeclaredThrowableException(var3);
        }
 }

这里调用自己定义的invoke方法。逻辑流程与下图图一致
在这里插入图片描述

代理对象也重写了:

 public final boolean equals(Object var1) throws  {
        try {
            return (Boolean)super.h.invoke(this, m1, new Object[]{var1});
        } catch (RuntimeException | Error var3) {
            throw var3;
        } catch (Throwable var4) {
            throw new UndeclaredThrowableException(var4);
        }
    }

    public final String toString() throws  {
        try {
            return (String)super.h.invoke(this, m2, (Object[])null);
        } catch (RuntimeException | Error var2) {
            throw var2;
        } catch (Throwable var3) {
            throw new UndeclaredThrowableException(var3);
        }
    }
      public final int hashCode() throws  {
        try {
            return (Integer)super.h.invoke(this, m0, (Object[])null);
        } catch (RuntimeException | Error var2) {
            throw var2;
        } catch (Throwable var3) {
            throw new UndeclaredThrowableException(var3);
        }
    }

该方法不可重写,针对执行结果进行不同处理。

2.2 CGLIb代理

public class CGLIBProxy  {

    @SuppressWarnings("unchecked")
    public static <T> T getProxy(final Object object) {
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(object.getClass());
        enhancer.setCallback(new MethodInterceptor() {
            @Override
            public Object intercept(Object o, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
                return  method.invoke(object, args);
            }
        });

        return (T) enhancer.create();
    }
}

3.小结

  • 代理模式(Proxy Pattern)中,一个类代表另一个类的功能;在访问此对象时加上一个对此对象的访问层,控制对代理对象的访问。
  • 优点:高扩展性。缺点:处理速度慢。
  • 静态代理,代理类在编译时生成
  • 代理类在程序运行时创建的代理。
  • JDK动态代理创建,需要接口,CGLIb动态代理创建不需要接口。
  • CGLIb动态代理缺陷:被代理对象final方法不可被重写。

参考:菜鸟教程:代理模式

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值