Java基础之让你彻底搞懂代理模式

 更新时间:2021年04月26日 15:38:00   作者:猿华  
这篇文章主要介绍了Java基础之让你彻底搞懂代理模式,文中有非常详细的代码示例,对正在学习java基础的小伙伴们有非常好的帮助,需要的朋友可以参考下

一、代理模式

什么是代理模式?

先来生活常用例子:你想买票,你没必要去车站买;而是可以去一个代售点,代售点代理车站卖票,这就是一个简单的代理模式!

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

二、静态代理

顾名思义,静态的,由程序员构写,在编译是就已经将程序接口,代理类和被代理类写定了!在程序运行前就已经生成!
来个简单实例:

在这里插入图片描述

设计公共接口Person:

public interface Person {
    public void handinWork();
}

Student:

public class Student implements Person {
    private String name;

    public Student(String name) {
        this.name=name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public void handinWork() {
        System.out.println(name+"提交作业");
    }
}

代理类StudentProxy:

public class StudentProxy implements Person {
    //代理谁
    Student student;
    //防止重名,确保只是代理Student这个对象
    public StudentProxy(Student student) {
        if(student.getClass()==Student.class){
            this.student = (Student) student;
        }
    }

    public void handinWork() {
        //交作业时代理(班长)想代替学生搞点事···
        System.out.println("老师这个比的作业是抄的呢");
        student.handinWork();
    }
}

测试:

public class StaticProxy {
    public static void main(String[] args) {
        //创建出两个对象,代理和被代理
        Student student = new Student("张三");
        StudentProxy monitor = new StudentProxy(student);
        //代理类去交作业!!!!!
        monitor.handinWork();
    }
}

可以发现,程序本质还是学生交作业,不过是通过一个中间层monitor(班长)去实现的,然后这个代理层还可以添加些其他的操作功能,在提交作业之前或者之后!

三、动态代理

静态代理是在程序运行前就生成了,很明显动态代理就是在程序运行期间添加代理层以达到效果。代理类动态生成!

  • 需要了解两个类:Proxy 代理, InvocationHandler
  • Proxy:代理类,使用的时候动态生成
  • InvocationHandler:主要执行需要代理的方法,使用invoke执行

举个例子:方便理解,就上面那个实例我们将它改造成一个动态的!

第一步,创建一个类实现InvocationHandler接口,用它构建出代理类和代理方法:

public class ProxyInvocation implements InvocationHandler {
    //被代理的接口,真实的对象
    private Person person;
    //生成得到的代理类

    public void setPerson(Person person) {
        this.person = person;
    }
    //通过Proxy类的newProxyInstance方法创建代理对象,我们来看下方法中的参数
    //         * 第一个参数:people.getClass().getClassLoader(),使用handler对象的classloader对象来加载我们的代理对象
    //         * 第二个参数:people.getClass().getInterfaces(),这里为代理类提供的接口是真实对象实现的接口,这样代理对象就能像真实对象一样调用接口中的所有方法
    //         * 第三个参数:handler,我们将代理对象关联到上面的InvocationHandler对象上
    public Object getProxy(){
        return Proxy.newProxyInstance(this.getClass().getClassLoader(),person.getClass().getInterfaces(),this);
    }
    //处理代理实例,返回结果
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        Object result = method.invoke(person, args);
        return result;
    }
}

参照静态代理,需要一个代理对象,只不过静态代理中代理对象是我们手动创建的,所以动态代理中我们利用jdk自带代理类Proxy的newProxyInstance方法动态的生成了一个代理类而已,这个代理类可以根据我们传的参数动态改变,

测试:

 //真实角色student
        Student student = new Student("张三");
        //拿到InvocationHandler的继承类
        ProxyInvocation pih = new ProxyInvocation();
        //拿到代理的接口
        pih.setPerson(student);
        //动态生成了proxy代理类
        Person proxy = (Person) pih.getProxy();
        //代理类执行方法
        proxy.handinWork();

改进成工具类:将获得接口部分全部采用参数代替;

private Object target;
    //生成得到的代理类

    public void setPerson(Object target) {
        this.target= target;
    }
    //通过Proxy类的newProxyInstance方法创建代理对象,我们来看下方法中的参数
    //         * 第一个参数:people.getClass().getClassLoader(),使用handler对象的classloader对象来加载我们的代理对象
    //         * 第二个参数:people.getClass().getInterfaces(),这里为代理类提供的接口是真实对象实现的接口,这样代理对象就能像真实对象一样调用接口中的所有方法
    //         * 第三个参数:handler,我们将代理对象关联到上面的InvocationHandler对象上
    public Object getProxy(){
        return Proxy.newProxyInstance(this.getClass().getClassLoader(),target.getClass().getInterfaces(),this);
    }
    //处理代理实例,返回结果
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        Object result = method.invoke(person, args);
        return result;
    }

就是把接口类型和参数改一下就可以!后续会添加代理模式在Spring上的使用,

四、总结

你不敲一遍代码你永远都不会理解,你抱着学一遍而不是彻底弄明白你也不可能学会!多查资料多理解代码,多花点时间一定会懂得!加油老铁。以解决问题的心态学编程而不是为了高薪工作。

到此这篇关于Java基础之让你彻底搞懂代理模式的文章就介绍到这了,更多相关java代理模式内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • springboot实现rabbitmq消息确认的示例代码

    springboot实现rabbitmq消息确认的示例代码

    RabbitMQ的消息确认有两种, 一种是消息发送确认,第二种是消费接收确认,本文主要介绍了springboot实现rabbitmq消息确认的示例代码,具有一定的参考价值,感兴趣的可以了解一下
    2023-09-09
  • spring boot下 500 404 错误页面处理的方法

    spring boot下 500 404 错误页面处理的方法

    本篇文章主要介绍了spring boot下 500 404 错误页面处理的方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-04-04
  • Java 操作Properties配置文件详解

    Java 操作Properties配置文件详解

    本篇文章主要介绍了Java 操作Properties配置文件详解,详细的介绍了Properties和主要方法,有兴趣的可以了解下
    2017-05-05
  • 深入浅析jni中的java接口使用

    深入浅析jni中的java接口使用

    这篇文章主要介绍了jni中的java接口使用,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-09-09
  • Java 进程执行外部程序造成阻塞的一种原因

    Java 进程执行外部程序造成阻塞的一种原因

    前一阵子在研究文档展示时使用了java进程直接调用外部程序,其中遇到一个问题花了好长时间才解决,这个问题就是外部程序直接执行没什么问题,但是当使用Java进程执行时外部程序就阻塞在那儿不动了。而且这个外部程序在处理某些文件时使用Java进程执行是没问题的
    2014-03-03
  • 解决nacos启动报错Server check fail, please check server localhost ,port 9848 is available的问题

    解决nacos启动报错Server check fail, please che

    这篇文章主要介绍了nacos启动 Server check fail, please check server localhost ,port 9848 is available的错误原因以及解决方法,需要的朋友可以参考下
    2023-09-09
  • Java Spring事务使用及验证过程详解

    Java Spring事务使用及验证过程详解

    这篇文章主要介绍了Java Spring事务使用及验证过程详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-12-12
  • Java 数据结构与算法系列精讲之队列

    Java 数据结构与算法系列精讲之队列

    这篇文章主要介绍了Java队列数据结构的实现,队列是一种特殊的线性表,只允许在表的队头进行删除操作,在表的后端进行插入操作,队列是一个有序表先进先出,想了解更多相关资料的小伙伴可以参考下面文章的详细内容
    2022-02-02
  • Jmeter多台机器并发请求实现压力性能测试

    Jmeter多台机器并发请求实现压力性能测试

    这篇文章主要介绍了Jmeter多台机器并发请求实现压力性能测试,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-10-10
  • Spring中的父子容器原理解析

    Spring中的父子容器原理解析

    这篇文章主要为大家介绍了Spring中的父子容器原理解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-07-07

最新评论