JAVA 代理模式

文章目录

介绍

代理模式

静态代理

动态代理

Cglib代理

介绍

描述

代理模式设计模式属于结构型模式,为当前类对象提供一个替身(一个类代表另一个类的功能),通过替身来控制对这个对象的访问,这样我们就可以在目标对象调用的基础上增加额外的功能。

特点

优点:拓展性强、能够协调调用者和被调用者。

缺点:中间增加的代理对象可能导致请求处理速度的变慢、实现代理的额外根据其不同方式可能实现复杂。

实现原理

增加中间层去取代直接调用的方式,而代理类实现与目标对象组合。

适用

  1. 应用远程代理控制访问远程对象。

  2. 保护代理基于权限控制对资源的访问。

  3. 虚拟代理控制访问创建开销大的对象。

  4. 动态代理等其他代理场景。

区别

与适配器模式的区别:适配器模式主要将一个现存的接口适配为期待的另一个接口,而代理模式不能改变所代理类的接口。

与装饰器模式的区别:装饰器模式为了增强当前对象的功能,而代理模式是为了加以控制对当前对象的使用。

代理模式

静态代理

创建一个书籍IBook接口,提供一个借书方法borrow(),将其由类Book实现,提供代理类BookProxy,实现同样的IBook接口,并设置实际调用成员。

条件:静态代理中代理对象和目标对象需要实现接口。

// 接口
public interface IBook {
    void borrow();
}
// 实现者
public class Book implements IBook{

    @Override
    public void borrow() {
        System.out.println("借出  x 本书");
    }
}

// 代理者
public class BookProxy implements IBook{

    private IBook target;


    public BookProxy() {
    }


    @Override
    public void borrow() {

        if (target == null){
            this.target = new Book();
        }

        System.out.println("XXX人 代理中...");
        target.borrow();
        System.out.println("XXX人 代理结束...");
    }
}
// 测试
public class Test {
    public static void main(String[] args) {
        // 创建代理对象
        BookProxy proxy =  new BookProxy();
        // 代理执行
        proxy.borrow();
        /*
           XXX人 代理中...
           借出  x 本书
          XXX人 代理结束...*/
    }
}

动态代理

目标对象必须实现接口,代理对象的生成是通过JDK的API进行的,动态的在内存中构建代理对象。

条件:目标对象必须实现接口。


import java.lang.reflect.Proxy;

public class ProxyFactory {
    // 创建一个 Object类型的目标对象
    private Object target;

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

    public Object getProxyInstance() {
        /**
         *  public static Object newProxyInstance(
         *    ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)
         *
         * 参数:
         *  loader:指定 目标对象 的类加载器,获取加载器的方法固定
         *  interface:目标对象实现的接口类型
         *  h: 事件处理,当代理对象 执行 该目标对象的方法 时,触发该事件处理器中方法,
         *  通过将 目标对象的方法作为参数传入  并且将执行结果返回。
         */
        return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(),
                (proxy, method, args) -> {
                    System.out.println("动态代理中...");
                    // 获取反射机制调用的目标方法
                    Object rVal = method.invoke(target,args); // 目标对象  + args参数 返回执行结果
                    return rVal;
                });
    }
}

// 测试
public class Test {
    public static void main(String[] args) {
        // proxyInstance 为内存中动态生成的代理对象
        IBook proxyInstance = (IBook) new ProxyFactory(new Book()).getProxyInstance();
        // System.out.println(proxyInstance.getClass());
        proxyInstance.borrow();
        /*
        class com.sun.proxy.$Proxy0
        动态代理中...
        借出  x 本书
        */
    }
}

Cglib代理

当有时候目标对象只是一个单独对象未实现任何目标接口,上述两种就不可使用,转而使用cglib代理即可解决,同样的在内存中构建一个子类对象来实现对目标对象的拓展。

条件:目标对象不需要实现接口时可以使用该代理、被代理的类不能够被final修饰、被代理的方法如果被static和final修饰将不会被拦截。

导包:cglib-2.2.jar、asm-tree.jar、asm-commons.jar、asm.jar。

// 目标对象
public class Book {

    public void borrow() {
        System.out.println("借出  x 本书");
    }
}
// 代理
// 实现 cglib中的 MethodInterceptor 拦截接口 重写intercept
public class ProxyFactory implements MethodInterceptor {
    // 创建一个 Object类型的目标对象
    private Object target;

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

    public Object getProxyInstance(){
        Enhancer enhancer = new Enhancer(); // cglib增强类
        enhancer.setSuperclass(target.getClass());  // 设置目标类
        enhancer.setCallback(this); // 设置委托对象
        return enhancer.create();   // 创建代理对象
    }

    @Override
    public Object intercept(Object o, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
        System.out.println("cglib代理 开始....");
        Object rVal = method.invoke(target, args);
        System.out.println("cglib代理 结束....");
        return rVal;
    }
}
//测试
public class Test {
    public static void main(String[] args) {

        // proxyInstance 为内存中动态生成的代理对象
        IBook proxyInstance = (IBook) new ProxyFactory(new Book()).getProxyInstance();
        System.out.println(proxyInstance.getClass());

        proxyInstance.borrow();
        /*
        class com.sheji.proxy.Book$$EnhancerByCGLIB$$44466c5a
        cglib代理 开始....
        借出  x 本书
        cglib代理 结束....
        */
    }
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值