我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Spring细数两种代理模式之静态代理和动态代理概念及使用

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Spring细数两种代理模式之静态代理和动态代理概念及使用

代理模式

在学习 AOP 之前,我们先来了解下代理模式, 代理模式分为静态代理以及动态代理,属于23中设计模式之一。 AOP 的底层机制就是动态代理。

作用:

通过代理类为原始类增加额外的功能

1、静态代理

1)案例展示

【1】定义一个 Teacher 接口,接口中定义两个方法:teachOnLine 和 teachOffLine。

package cn.gc.spring03.interfce;
public interface Teacher {
    String teachOnLine(String course);
    String teachOffLine(String course);
}

【2】定义一个实现 Teacher 接口的 TeacherA 实现类,代码如下:

package cn.gc.spring03.interfce.impl;
import cn.gc.spring03.interfce.Teacher;
public class TeacherA implements Teacher {
    @Override
    public String teachOnLine(String course) {
        System.out.println("开始"+course+"课程线上教学");
        return course+"课程线上教学";
    }
    @Override
    public String teachOffLine(String course) {
        System.out.println("开始"+course+"课程线下教学");
        return course+"课程线下教学";
    }
}

【3】现在我们要在两个方法中的 开始课程教学 的前后添加上下课铃声的功能,此时我们再定义一个实现 Teacher 接口的静态代理类 TeacherAStaticProxy,代码如下:

package cn.gc.spring03.proxy;
import cn.gc.spring03.interfce.Teacher;
import cn.gc.spring03.interfce.impl.TeacherA;
public class TeacherAStaticProxy implements Teacher {
    TeacherA teacherA=new TeacherA();
    @Override
    public String teachOnLine(String course) {
        System.out.println(course+"课程上课时间到了");
        String s = teacherA.teachOnLine(course);
        System.out.println(course+"课程下课时间到了");
        return s;
    }
    @Override
    public String teachOffLine(String course) {
        System.out.println(course+"课程上课时间到了");
        String s = teacherA.teachOffLine(course);
        System.out.println(course+"课程下课时间到了");
        return s;
    }
}

【4】运行效果如下

2)静态代理优缺点

(1)优点 :在不修改目标对象的功能前提下,能通过代理对象对目标功能扩展;可以使得真实角色更加纯粹,不再去关注一些公共的事情;公共的业务由代理来完成,实现了业务的分工。

(2)缺点 :因为代理对象需要与目标对象实现一样的接口,所以会很多代理类 ,一旦接口增加方法,目标对象与代理对象都要维护,工作量变大,开发效率降低。

3)开发代理对象的原则

  • 代理对象和目标对象实现相同的接口
  • 代理对象依赖于目标对象

2、动态代理

在不改变原来的代码的情况下,实现了对原有功能的增强,这是 AOP 中最核心的思想。

AOP:纵向开发,横向开发

1)简介

动态代理的角色和静态代理的一样。但是动态代理的代理类是动态生成的,而静态代理的代理类是提前写好的。

动态代理分为两类,一类是基于接口动态代理(JDK),一类是基于类的动态代理(CGLib)。

2)基于接口动态代理(JDK)

程序运行的过程中,通过 JDK 提供代理技术动态的为某个类产生动态代理对象的过程。

开发代理对象的原则:

  • 代理对象,不需要实现接口,但是目标对象要实现接口,否则不能用JDK动态代理。
  • 代理对象的生成,是利用JDK的API,动态的在内存中构建代理对象。
  • 动态代理也叫做 :JDK代理、接口代理。

JDK 的动态代理需要了解两个类:

核心 : InvocationHandler (调用处理程序) 和 Proxy (代理)

实现步骤:在上面案例的基础上实现。

创建一个基于 JDK 的代理工具类 DynamicProxy 类:

package cn.gc.spring03.proxy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

public class DynamicProxy {
    //被代理对象
    private Object target;
    public DynamicProxy(Object obj) {
        this.target = obj;
    }
    //生成代理对象
    public Object getProxy() {
        //获取类加载器
        ClassLoader classLoader = target.getClass().getClassLoader();
        //获取被代理对象实现的所有接口
        Class<?>[] interfaces = target.getClass().getInterfaces();
        //增强功能代码编写的位置
        InvocationHandler invocationHandler = new InvocationHandler() {
            @Override
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                // 增强的功能
                System.out.println(args[0] + "课程上课时间到了");
                // 通过反射执行目标方法
                Object invoke = method.invoke(target, args);
                // 增强的功能
                System.out.println(args[0] + "课程下课时间到了");
                return invoke;
            }
        };
        Object o = Proxy.newProxyInstance(classLoader, interfaces, invocationHandler);
        return o;
    }
}

测试:

    @Test
    public void test02() {
        Teacher teacher  = new TeacherA();
        DynamicProxy dynamicProxy = new DynamicProxy(teacher);
        Teacher teacherProxy = (Teacher) dynamicProxy.getProxy();
        teacherProxy.teachOnLine("Java");
        System.out.println("______________________");
        teacherProxy.teachOnLine("C++");
    }

运行效果:

3)基于类的动态代理(CGLib)

开发代理对象的原则:

  • 代理对象无需和原始类对象实现相同的接口
  • 代理对象和原始类对象要存在父子类关系

CGLib 的动态代理需要了解两个类:

  • 核心 : Enhancer 和 MethodInterceptor

实现步骤

创建一个 Aoo 类:

package cn.gc.spring03.interfce.impl;
public class Aoo {
    public String test01(String info){
        System.out.println("Aoo中的test01方法被调用。");
        return info;
    }
}

创建一个基于 CGLib 的代理工具类 CglibProxy 类:

package cn.gc.spring03.proxy;
import org.springframework.cglib.proxy.Callback;
import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;
import java.lang.reflect.Method;
public class CglibProxy {
    private Object target;
    public CglibProxy(Object obj){
        this.target=obj;
    }
    public Object getProxy(){
        Enhancer enhancer=new Enhancer();
        //设置类加载器
        enhancer.setClassLoader(CglibProxy.class.getClassLoader());
        //设置被代理对象
        enhancer.setSuperclass(target.getClass());
        //增强功能代码编写的位置
        MethodInterceptor methodInterceptor=new MethodInterceptor() {
            @Override
            public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
                System.out.println("-------------cglib log--------------");
                Object invoke = method.invoke(target, objects);
                System.out.println("-------------cglib log--------------");
                return invoke;
            }
        };
        enhancer.setCallback(methodInterceptor);
        //代理对象
        Object o = enhancer.create();
        return o;
    }
}

测试:

    @Test
    public void test03() {
        Aoo aoo=new Aoo();
        CglibProxy cglibProxy = new CglibProxy(aoo);
        Aoo proxy = (Aoo) cglibProxy.getProxy();
        String info = proxy.test01("java");
        System.out.println("info = " + info);
    }

运行效果:

4)动态代理的优势

  • 可以使得真实角色更加纯粹,不再去关注一些公共的事情。
  • 公共的业务由代理来完成,实现了业务的分工。
  • 公共业务发生扩展时变得更加集中和方便。
  • 一个动态代理,一般代理某一类业务。
  • 一个动态代理可以代理多个类,代理的是接口。

到此这篇关于Spring细数两种代理模式之静态代理和动态代理概念及使用的文章就介绍到这了,更多相关Spring静态代理与动态代理内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Spring细数两种代理模式之静态代理和动态代理概念及使用

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Spring细数两种代理模式之静态代理和动态代理概念及使用

代理是一种设计模式,提供了对目标对象另外的访问方式,即通过代理对象访问目标对象。可以不修改目标对象,对目标对象功能进行拓展。在我们学习Spring的时候就会发现,AOP(面向切面编程)的底层就是代理
2023-02-27

Spring静态代理和动态代理概念及使用方法是什么

这篇文章主要讲解了“Spring静态代理和动态代理概念及使用方法是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Spring静态代理和动态代理概念及使用方法是什么”吧!代理模式在学习 A
2023-07-05

Java代理模式与动态代理之间的关系以及概念

代理模式是开发中常见的一种设计模式,使用代理模式可以很好的对程序进行横向扩展。动态代理:代理类在程序运行时被创建的代理方式。关键在于动态,程序具有了动态特性,可以在运行期间根据不同的目标对象生成动态代理对象
2023-02-20

Spring之AOP两种代理机制对比分析(JDK和CGLib动态代理)

这篇文章主要介绍了Spring之AOP两种代理机制对比分析(JDK和CGLib动态代理),具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
2023-05-19

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录