Java--代理设计模式

所谓代理设计模式,简单的是说,就是不改变源代码的情况下,对目标对象进行功能扩展

静态代理

下面以一个简单的案例说明:

需求:有个歌手对象Singer,该对象有个唱歌方法sing(),用代理设计模式实现唱歌前后向观众问好和致谢(功能扩展)

步骤:定义SingerImp接口;定义Singer类,并实现SingerImp接口;定义ProxySinger类,并实现SingerImp接口。

public interface SingerImp {
    void sing();
}
public class Singer implements SingerImp {
    @Override
    public void sing() {
        System.out.println("唱一首歌");
    }
}
public class SingerProxy implements SingerImp {
    private Singer target;

    public SingerProxy(Singer target) {
        this.target = target;
    }

    @Override
    public void sing() {
        System.out.println("向观众问好");
        target.sing();
        System.out.println("谢谢大家");
    }
}

测试类

class SingerProxyTest {
    public static void main(String[] args) {
        Singer singer = new Singer();
        SingerProxy singerProxy = new SingerProxy(singer);
        singerProxy.sing();
    }
}

总结:这种实现方式很直观也很简单,但是缺点是代理对象必须提前写出。如果接口层发生了变化,代理对象的代码也要进行维护。

动态代理

定义一个接口,JDk自动给这个接口生成一个实现类,在这个实现类中,方法的执行逻辑是java.lang.reflect.InvocationHandler # invoke(java.lang.Object, java.lang.reflect.Method, java.lang.Object[]),关于这三个参数的说明,请看注解。

public interface SingerImp {
    void sing();
}
public class Singer implements SingerImp {
    @Override
    public void sing() {
        System.out.println("唱一首歌");
    }
}

测试类

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

class SingerTest {
    public static void main(String[] args) {
        Singer singer = new Singer();
        //当前目标对象使用的类加载器
        ClassLoader loader = singer.getClass().getClassLoader();
        //目标对象实现的接口类型
        Class<?>[] interfaces = singer.getClass().getInterfaces();
        //功能扩展,核心部分
        InvocationHandler handler = new InvocationHandler() {
            /**
             * @param o 代理对象本身
             * @param method 被代理对象的所有方法,对于目前这个案例来说,就是sing()
             * @param objects 方法的参数
             * @return JDk自动生成的实现类
             */
            @Override
            public Object invoke(Object o, Method method, Object[] objects) throws Throwable {
                System.out.println("向观众问好");
                Object invoke = method.invoke(singer, objects);
                System.out.println("谢谢大家");
                return invoke;
            }
        };
        SingerImp singerImp = (SingerImp) Proxy.newProxyInstance(loader, interfaces, handler);
        singerImp.sing();
    }
}

总结:静态代理和动态代理都需要定义一个接口

Cglib代理

前提条件:

步骤:定义Singer类;定义代理 工厂类ProxyFactory;测试

public class Singer {
    public void sing() {
        System.out.println("唱一首歌");
    }
}
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;

import java.lang.reflect.Method;

public class ProxyFactory implements MethodInterceptor {
    //目标对象
    private Object singer;

    public ProxyFactory(Object singer) {
        this.singer = singer;
    }

    //给目标对象创建一个代理对象
    public Object getProxyInstance() {
        //工具类
        Enhancer enhancer = new Enhancer();
        //设置父类
        enhancer.setSuperclass(singer.getClass());
        //设置回调函数
        enhancer.setCallback(this);
        //创建代理对象
        return enhancer.create();
    }

    @Override
    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        System.out.println("向观众问好");
        Object invoke = method.invoke(singer, objects);
        System.out.println("谢谢大家");
        return invoke;
    }
}

测试类

public class Test01 {
    public static void main(String[] args) {
        //目标对象
        Singer singer = new Singer();
        //代理对象
        Singer proxy = (Singer) new ProxyFactory(singer).getProxyInstance();
        //执行代理对象的方法
        proxy.sing();
    }
}

总结:

JDK动态代理:(优势)官方提供的,剋直接使用,不需要添加依赖,效率高;(劣势)只支持有接口的类;

CGLIB动态代理:(优势)无论类是否有接口,都支持通过cglib创建代理对象;(劣势)第三方的工具,使用时需要添加第三方依赖,效率低。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值