代理模式(十五)

相信自己,请一定要相信自己

上一章简单介绍了享元模式(十四), 如果没有看过, 请观看上一章

一. 代理模式

引用 菜鸟教程里面的代理模式介绍: https://www.runoob.com/design-pattern/proxy-pattern.html

在代理模式(Proxy Pattern)中,一个类代表另一个类的功能。这种类型的设计模式属于结构型模式。

在代理模式中,我们创建具有现有对象的对象,以便向外界提供功能接口。

一.一 介绍

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

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

何时使用: 想在访问一个类时做一些控制。

如何解决: 增加中间层。

关键代码: 实现与被代理类组合。

应用实例: 1、Windows 里面的快捷方式。 2、猪八戒去找高翠兰结果是孙悟空变的,可以这样理解:把高翠兰的外貌抽象出来,高翠兰本人和孙悟空都实现了这个接口,猪八戒访问高翠兰的时候看不出来这个是孙悟空,所以说孙悟空是高翠兰代理类。 3、买火车票不一定在火车站买,也可以去代售点。 4、一张支票或银行存单是账户中资金的代理。支票在市场交易中用来代替现金,并提供对签发人账号上资金的控制。 5、spring aop。

扫描二维码关注公众号,回复: 15541642 查看本文章

优点: 1、职责清晰。 2、高扩展性。 3、智能化。

缺点: 1、由于在客户端和真实主题之间增加了代理对象,因此有些类型的代理模式可能会造成请求的处理速度变慢。 2、实现代理模式需要额外的工作,有些代理模式的实现非常复杂。

使用场景: 按职责来划分,通常有以下使用场景: 1、远程代理。 2、虚拟代理。 3、Copy-on-Write 代理。 4、保护(Protect or Access)代理。 5、Cache代理。 6、防火墙(Firewall)代理。 7、同步化(Synchronization)代理。 8、智能引用(Smart Reference)代理。

注意事项: 1、和适配器模式的区别:适配器模式主要改变所考虑对象的接口,而代理模式不能改变所代理类的接口。 2、和装饰器模式的区别:装饰器模式为了增强功能,而代理模式是为了加以控制。

在这里插入图片描述

二. 代理模式

有 静态代理, 动态代理 (JDK代理) 和 Cglib 代理 ( 可以在内存动态的创建对象)

二.一 静态代理

二.一.一 教师讲课的接口

public interface ITeacher {
    
    

    public void talk();
}

二.一.二 一个教师讲课的接口实现

@Slf4j
public class RealTeacher implements ITeacher{
    
    

    @Override
    public void talk() {
    
    
        log.info("老师正在讲课");
    }
}

二.一.三 不使用代理

 @Test
    public void noProxyTest() {
    
    
        ITeacher iTeacher = new RealTeacher();
        iTeacher.talk();
    }

INFO [main] (RealTeacher.java:16) - 老师正在讲课

二.一.四 使用代理

@Slf4j
public class TeacherProxy implements ITeacher{
    
    
    private ITeacher iTeacher ;

    public TeacherProxy (ITeacher iTeacher) {
    
    
        this.iTeacher = iTeacher;
    }

    @Override
    public void talk() {
    
    
        log.info("老师提前备课--> 用户 prepare 操作");
        iTeacher.talk();
        log.info("老师课后总结--> 用户 after 操作");
    }

}

二.一.五 代理测试方法

@Test
    public void oneTest() {
    
    
        RealTeacher realTeacher = new RealTeacher();
        TeacherProxy teacherProxy = new TeacherProxy(realTeacher);
        teacherProxy.talk();
    }

image-20230615145810997

二.二 动态代理

二.二.一 接口

public interface ITeacher2 {
    
    

    public void talk();

    public void hello (String name);
}

二.二.二 接口实现

@Slf4j
public class RealTeacher2 implements ITeacher2{
    
    

    @Override
    public void talk() {
    
    
        log.info("老师正在讲课");
    }

    @Override
    public void hello(String name) {
    
    
        log.info("你好啊 {}", name);
    }
}

二.二.三 JDK 代理实现

@Slf4j
public class TeacherProxy2 {
    
    

    private Object target ;

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

    /**
     获取动态代理
     */
    public Object getProxyInstance() {
    
    
        return Proxy.newProxyInstance(
                target.getClass().getClassLoader(),
                target.getClass().getInterfaces(),
                new InvocationHandler() {
    
    
                    @Override
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    
    
                        log.info("JDK代理开始");

                        Object returnVal = method.invoke(target, args);

                        log.info("JDK代理结束");
                        return returnVal;
                    }
                }
        );
    }

}

二.二.四 代理测试

   @Test
    public void twoTest() {
    
    
        ITeacher2 teacher2 = new RealTeacher2();
        // 处理转换代理信息
        ITeacher2 target = (ITeacher2)(new TeacherProxy2(teacher2).getProxyInstance());
        // 进行处理
        target.hello("张三");
    }

image-20230615150102593

二.三 Cglib 代理

二.三.一 一个普通的类

@Slf4j
public class RealTeacher3 {
    
    

    public void talk() {
    
    
        log.info("老师正在讲课");
    }

    public void hello(String name) {
    
    
        log.info("你好啊 {}", name);
    }
}

二.三.二 对该类进行代理 TeacherProxy3

@Slf4j
public class TeacherProxy3 implements MethodInterceptor {
    
    
    private Object target;
    public TeacherProxy3(Object target) {
    
    
        this.target = target;
    }
    
    public Object getProxyInstance () {
    
    
        // 创建工具类 Enhancer
        Enhancer enhancer = new Enhancer();
        // 设置父类
        enhancer.setSuperclass(target.getClass());
        // 设置回调函数, 主要是这个。
        enhancer.setCallback(this);
        // 创建子类对象,即代理对象
        return enhancer.create();
    }
    
    @Override
    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
    
    
        log.info("CGLIB 代理开始");

        Object returnVal = methodProxy.invoke(target, objects);

        log.info("CGLIB 代理结束");
        return returnVal;
    }

}

二.三.三 代理测试

@Test
    public void threeTest() {
    
    
        RealTeacher3 realTeacher3 = new RealTeacher3();
        // 处理转换代理信息
        RealTeacher3 target = (RealTeacher3)new TeacherProxy3(realTeacher3).getProxyInstance();
        // 进行处理
        target.hello("张三");
    }

image-20230615150340055

三. Hutool 的 JDK 代理

可以使用 hutool 的 ProxyUtil ProxyFactory 进行代理

@Test
    public void fourTest() {
    
    
        ITeacher2 teacher2 = new RealTeacher2();

        ITeacher2 proxy = ProxyUtil.proxy(teacher2, new SimpleAspect());

        proxy.hello("hutool 工具类的代理");

        ITeacher2 proxy2 = ProxyFactory.createProxy(teacher2, new Aspect() {
    
    
            @Override
            public boolean before(Object target, Method method, Object[] args) {
    
    
                log.info(">>>> 代理之前执行的操作");
                return true;
            }

            @Override
            public boolean after(Object target, Method method, Object[] args, Object returnVal) {
    
    
                log.info(">>>> 代理之后执行的操作");
                return true;
            }

            @Override
            public boolean afterException(Object target, Method method, Object[] args, Throwable e) {
    
    
                log.info(">>>> 异常时执行的操作");
                return true;
            }
        });
        proxy2.hello("hutool 工厂的代理");
    }

image-20230615150904736

CGLIB创建的动态代理对象比JDK创建的动态代理对象的性能更高,但是CGLIB创建代理对象时所花费的时间却比JDK多得多。

所以对于单例的对象,因为无需频繁创建对象,用CGLIB合适,反之使用JDK方式要更为合适一些。

同时由于CGLib由于是采用动态创建子类的方法,对于final修饰的方法无法进行代理


本章节的代码放置在 github 上:


https://github.com/yuejianli/DesignPattern/tree/develop/Proxy


谢谢您的观看,如果喜欢,请关注我,再次感谢 !!!

猜你喜欢

转载自blog.csdn.net/yjltx1234csdn/article/details/131228558