23模式--代理模式

本文介绍了代理模式的概念和作用,通过代码示例展示了静态代理、JDK动态代理和Cglib代理的实现方式。静态代理需要代理类和目标类实现相同接口,动态代理则基于接口生成代理对象,而Cglib代理通过创建目标对象的子类来实现。文章还探讨了各种代理的优缺点及其适用场景。
摘要由CSDN通过智能技术生成

本篇主要聊一些23中模型中的代理模式:

看一下百度百科的解释:

代理模式的定义:为其他对象提供一种代理以控制对这个对象的访问。在某些情况下,一个对象不适合或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用。

代理模式其实有点像是:合作方—经纪人—明星这样关系。

合作方如果想要找明星合作,首先要找到经纪人,具体的谈判合同的事情,先和经纪人进行协商,最后达成合作。

其实代理模式有很多不同的形式,主要有三种:静态代理,动态代理(JDK代理,接口代理),Cglib代理。

一般的组成有:

  • 抽象角色:通过接口或抽象类生命真实角色实现的业务方法。
  • 代理角色:实现抽象角色,是真实角色的代理,通过真实角色的业务逻辑方法来实现抽象方法,并可以附加自己的操作。
  • 真实角色:实现抽象角色,定义真实角色所要实现的业务逻辑,供代理角色调用。

静态代理

静态代理在使用时,需要定义接口或者抽象类,也就是被代理对象(明星)与代理对象(经纪人)需要实现相同的接口或者是继承相同的父类。

现在进行代码演示:

  • 接口

    public interface BusinessInfa {
    //    签约方法
       void signContract();
    
    }
    
    
  • 被代理类

    public class Star implements BusinessInfa {
        @Override
        public void signContract() {
            System.out.println("我是大明星,我同意这份合同了");
        }
    }
    
    
  • 代理人

    public class Agent implements BusinessInfa{
    //    代理谁
        Star  star;
    
        public Agent(Star star) {
            this.star = star;
        }
    
        @Override
        public void signContract() {
    //        先联系明星经纪人
            System.out.println("你好,你找我家明星合作,可以和我谈");
    //        谈合同不是简单就可以签约的,肯定要涉及道各种拉扯,合同条款以及费用
            System.out.println("和经纪人一起疯狂的如果老太太菜市场砍价一般,深入几天各种约各种谈");
    //        最后同意了合同
            this.star.signContract();
    //        就算合约签了,具体后面合作中的事情
            System.out.println("具体合作后现场一些细节,比如我家大明星,剧本改下台词超过十个字了,记不住台词");
    
        }
    }
    
    
  • 客户端调用代理人

    public class client {
        public static void main(String[] args) {
            //    需要找大明星的联系方式没有,去找大明星公开的经纪人联系
            System.out.println("我是合作方,需要联系大明星,没办法只能先联系经纪人");
            BusinessInfa businessInfa=new Agent(new Star());
            businessInfa.signContract();
        }
    
    
    }
    
    

然后看一下结果:

在这里插入图片描述

现在看一下静态代理的优缺点

  • 优点: 在不修改目标的对象功能的前提下,通过代理对象对目标公共进行扩展,例子中不会对明星签约的行动进行修改,但是具体谈判细节,以及合作后的出现事故等投通过经纪人进行谈判,毕竟明星给了经纪人钱的。
  • 缺点:代理对象需要与目标的对象实现一样的接口,所以会有很多代理类,一旦接口增量了方法,目标对象和代理对象都要维护。也就是比如合作定义的行为,如果增加了明星和经纪人都需要增加行为。

静态代理其实是最方便理解代理这个原理的,而其它无论如何变化,都不能离开这个原理。

动态代理:jdk代理

动态代理类,是位于Java.lang.reflect包下类别的Interface InvocationHandler。其实也是通过反射实现的,所以代理对象不需要实现接口,但是目标对象要实现接口,否则不能用动态代理。而这是利用JDK的API实现的动态代理是在内存中构建代理对象的。

动态代理也被称之为JDK代理,接口代理。

还是老规矩直接用代码演示:

  • 被代理类接口

    //商务接口  毕竟经纪人代理明星的业务,也是要统一一下什么业务能代理
    public interface BusinessInfa {
    //    签约方法
       void signContract();
       void breakContract();
    
    }
    
    
  • 被代理类:

    public class Agent {
        //    代理谁
        Object target;
    
        public Agent(Object target) {
            this.target = target;
        }
    
    
    //     动态创建不同的代理对象
        public Object getProxyObject(){
            Object object=Proxy.newProxyInstance(this.target.getClass().getClassLoader(),
                    this.target.getClass().getInterfaces(),new myInvocationHandler());
    
            return object;
    
        }
       class myInvocationHandler implements InvocationHandler{
    
           @Override
    //        proxy 在其上调用方法的代理实例   method被代理的方法    代理方法中的参数args
           public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    // 对第一个参数好奇是什么
               System.out.println("=======================================");
    //           System.out.println(proxy); 这样打印会报错  不过可以看出第一个参数应该是代理对象 一般的时候没有什么用
               System.out.println(proxy.getClass());
               System.out.println("=======================================");
    
    //           因为要运行的方法没有返回值,所以不接受数据,通过return返回了 同时这里也是可以根据反射判断不同的方法,然后加入不同的逻辑
               if(method.getName()=="signContract" || method.getName().equals("signContract")){
                   System.out.println("你好,你找我家明星合作,可以和我谈");
                   method.invoke(target,args);
                   System.out.println("后期出现事情,你们必须改,因为我家明星最漂亮,哪怕是她的错你们会原谅的。");
                   //           这里可以返回所代理的类要运行的方法,不过因为我没有返回值,所以直接返回空
                   return null;
               }else if (method.getName()=="breakContract" || method.getName().equals("breakContract")){
                   System.out.println("你好,你们的错误");
                   method.invoke(target,args);
                   System.out.println("可恶的合作方,不理你们了");
                   //           这里可以返回所代理的类要运行的方法,不过因为我没有返回值,所以直接返回空
                   return null;
               }
    
               return null;
           }
       }
    }
    
  • 调用类

    public class client {
        public static void main(String[] args) {
            //    需要找大明星的联系方式没有,去找大明星公开的经纪人联系
            System.out.println("我是合作方,需要联系大明星,没办法只能先联系经纪人");
            Agent agent=  new Agent(new Star());
            BusinessInfa businessInfa = (BusinessInfa) agent.getProxyObject();
            businessInfa.signContract();
            System.out.println("*********************************************");
            businessInfa.breakContract();
        }
    }
    

    然后看一下输出结果:

在这里插入图片描述
可以看下类关系图:
在这里插入图片描述

jdk代理的优缺点:

  • 优点 :JDK原声动态代理时java原声支持的、不需要任何外部依赖。而且可以动态生成代理类,方不需要像静态代理哪里代理类因为接口变化而不停的调整。
  • 缺点:但是它只能基于接口进行代理,也就是被代理的对象也需要有一个接口,不然无法使用jdk代理,同时因为它已经继承了proxy了,java不支持多继承。

动态代理:Cglib代理

无论静态代理还是上面提到的JDK动态代理都需要实现一个接口,但是有时候对象只是一个单独的对象,并没有实现任何的接口,这个时候就需要使用目标对象的子类来实现。而聊到的Cglib动态代理就算通过这种方式实现代理的。

Cglib代理也叫做子类代理,其是再内存中构建了一个子类对象,从而实现对目标对象功能的扩展。Cglib代理是可以再运行期扩展java类与实现java接口,所以其广泛被需要AOP框架使用,其中就包括spring,通过Cglib实现方法拦截。

因为是内存中动态构建子类,所以Cglib代理类不能为final。同样如果目标对象的方法如果为final或者static,代理也会不对其方法进行代理。

其实AOP中不一定会都选择使用Cglib代理,我们开发中同样是如此选择的:

  • 目标对象需要实现接口,那就使用JDK代理。
  • 目标对象不需要实现接口,那句使用Cglib代理。

Cglib包的底层是通过使用字节码处理框架ASM来转换字节码并生成新的类。

需要Jar包

这个有两种情况,

  • 如果引入的是Cglib的jar包就需要四个包:

    asm.jar asm-commons.jar asm-tree.jar Cglib-*.*.jar(自己选版本)

  • 如果使用 cglib-nodep的jar,直接导入这一个就行,因为其打包了cglib所需要的依赖jar包

代码演示

  • 被代理的对象,无需实现接口

    public class Star  {
    
        public void signContract() {
            System.out.println("我是大明星,我同意这份合同了");
        }
    
    
        public void breakContract() {
            System.out.println("台词超过三句了,编辑不修改剧本,合作方的错误,毁约了。。。。");
        }
    }
    
    
  • 使用Cglib代理的增强代理类

    // 经纪人代理  需要实现Cglib代理中接口方法MethodInterceptor
    public class Agent implements MethodInterceptor {
        //    代理谁
        Object target;
    
        public Agent(Object target) {
            this.target = target;
        }
    
    
    //     动态创建不同的代理对象
        public Object getProxyObject(){
    //        创建一个Cglib包下的工具栏
            Enhancer enhancer=new Enhancer();
    //        设置父类
            enhancer.setSuperclass(target.getClass());
    //        设置回调函数,这个回调本身是自己所以
            enhancer.setCallback((Callback) this);
    // 创建子类对象 也就是代理对象
            return enhancer.create();
    
        }
    
        @Override
    //    需要重写这个方法,代理类调用方法的时候会走个方法
    //     o 代表的this 就算代理增强的对象  Method 被代理对象执行的方法,也就是拦截的方法   objects 方法的参数   methodProxy 用于调用super(非拦截方法);可以根据需要调用多次
        public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
            //           因为要运行的方法没有返回值,所以不接受数据,通过return返回了 同时这里也是可以根据反射判断不同的方法,然后加入不同的逻辑
            if(method.getName()=="signContract" || method.getName().equals("signContract")){
                System.out.println("你好,你找我家明星合作,可以和我谈");
                method.invoke(target,objects);
                System.out.println("后期出现事情,你们必须改,因为我家明星最漂亮,哪怕是她的错你们会原谅的。");
                //           这里可以返回所代理的类要运行的方法,不过因为我没有返回值,所以直接返回空
                return null;
            }else if (method.getName()=="breakContract" || method.getName().equals("breakContract")){
                System.out.println("你好,你们的错误");
                method.invoke(target,objects);
                System.out.println("可恶的合作方,不理你们了");
                //           这里可以返回所代理的类要运行的方法,不过因为我没有返回值,所以直接返回空
                return null;
            }
    
            return null;
        }
    }
    
  • 调用测试的类

    public class client {
        public static void main(String[] args) {
            //    需要找大明星的联系方式没有,去找大明星公开的经纪人联系
            System.out.println("我是合作方,需要联系大明星,没办法只能先联系经纪人");
            Agent agent=  new Agent(new Star());
            Star star = (Star) agent.getProxyObject();
            star.signContract();
            System.out.println("*********************************************");
            star.breakContract();
        }
    }
    
    

在这里插入图片描述

也没有问题,可以实现动态代理。

补充 其它代理

这个只是写了几个代理的名字,也不是全部。

  • 防火墙代理: 内网通过代理穿透防火墙,对公网进行访问。
  • 缓存代理: 如果获取网络资源有些从缓存中回去资源,如果没有了再从其它地方获取资源。
  • 远程代理:远程代理通过网络和真正的远程对象沟通信息。常见的翻墙梯子就算这个逻辑。
  • 同步代理: 主要使用在多线程编程中,完成多线程间同步工作
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值