代理模式Proxy

一、代理模式(Proxy)
1.代理模式的定义

代理模式给某一个对象提供一个代理对象,并由代理对象控制对真实对象的访问,起到对代理对象已有功能的增强

通俗的来讲代理模式就是我们生活中常见的中介。

2.作用
  • **中介隔离作用:**在某些情况下,一个客户类不想或者不能直接引用一个委托对象,

而代理类对象可以在客户类和委托对象之间起到中介的作用,其特征是代理类和委托类实现相同的接口。

  • 开闭原则,增加功能:代理类除了是客户类和委托类的中介之外,我们还可以通过给代理

类增加额外的功能来扩展委托类的功能,这样做我们只需要修改代理类而不需要再修改委托类,符合代码设计的开闭原则。

3.代理模式的分类

我们有多种不同的方式来实现代理。如果按照代理创建的时期来进行分类的话可以分为两种:

  • 静态代理:

静态代理是由程序员创建或特定工具自动生成源代码,在对其编译。在程序员运行之前,代理类.class文件就已经被创建了。

  • 动态代理:

动态代理是在程序运行时通过反射机制动态创建的。

    • 动态代理分为:

基于接口的动态代理(jdk自带)

基于子类的动态代理(第三方)

二、静态代理

所谓"静态 "也就是在程序运行前就已经存在代理类的字节码文件,代理类和被代理对象的关系在运行前就确定了。

1.定义接口

复制代码
interface ISinger{
    public void sing();
}

2.定义被代理类

复制代码
public class SingerImp implements Isinger{
    @Override
    public void sing(){
        Systm.out.println("===莫等闲===");
    }
}

3.定义代理类

复制代码
public class SingerProxy implements Isinger{
    Isinger realsinger;
    
    public SingerProxy(Isinger realsinger) {
        this.realsinger = realsinger;
    }
    
    @Override
    public void sing(){
        Systm.out.println("大家好");
        realsinger.sing();
        Systm.out.println("谢谢");
    }
}

4.定义客户端,查看执行结果

复制代码
public class Test {
    public static void main(String[] args) {
        //1.创建被代理对象
        Isinger realsinger = new SingerImp();
        //2.创建代理
        Isinger proxysinger = new SingerProxy(realsinger);

        proxysinger.sing();
    }
}

输出结果:

大家好

莫等闲

谢谢

从上面的代码可以看出,静态代理存在被代理类和代理类之间耦合度过高。

三、动态代理

"动态"代理的源码是在程序运行期间由 JVM 根据反射等机制动态的生成,所以在运行前并不存在代理类的字节码文件。

1.使用JDk

1.1 定义接口

复制代码
interface ISinger{
    public void sing();
}

1.2 定义被代理类

复制代码
public class ZhaoYiTing implements ISinger{
    @Override
    public void sing() {
        System.out.println("===易燃易爆炸===");
    }
}

1.3 定义客户端,查看执行结果

复制代码
public class test01 {
    public static void main(String[] args) {
        //1.创建被代理对象
        ISinger zhaoYiTing = new ZhaoYiTing();
        //2.创建代理对象
        ISinger jinjiren = (ISinger)Proxy.newProxyInstance(zhaoYiTing.getClass().getClassLoader(), zhaoYiTing.getClass().getInterfaces(), new InvocationHandler() {
            @Override
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                //功能增强
                System.out.println("===自我介绍===");

                Object obj = method.invoke(zhaoYiTing, args);
                return obj;
            }
        });
        jinjiren.sing();
    }
}

输出结果:

===自我介绍===

===易燃易爆炸===

2.使用第三方(CGLIB)

2.1 定义接口

复制代码
interface ISinger{
    public void sing();
}

2.2 定义被代理类

复制代码
public class ZhaoYiTing implements ISinger{
    @Override
    public void sing() {
        System.out.println("莫等闲");
    }
}

2.3 定义客户端,查看执行结果

复制代码
public class test {
    public static void main(String[] args) {
        ISing imp = new tenggeerImp();
        ISing jin = (ISing)Enhancer.create(imp.getClass(), imp.getClass().getInterfaces(), new InvocationHandler() {
            @Override
            public Object invoke(Object o, Method method, Object[] objects) throws Throwable {
                Object obj = method.invoke(imp, objects);
                return obj;
            }
        });
        jin.sing();
    }
}

输出结果:

莫等闲

相关推荐
二哈赛车手8 小时前
新人笔记---ApiFox的一些常见使用出错
java·笔记·spring
栗子~~9 小时前
JAVA - 二层缓存设计(本地缓冲+redis缓冲+广播所有本地缓冲失效) demo
java·redis·缓存
YDS8299 小时前
DeepSeek RAG&MCP + Agent智能体项目 —— RAG知识库的搭建和接口实现
java·ai·springboot·agent·rag·deepseek
未若君雅裁10 小时前
MyBatis 一级缓存、二级缓存与清理机制
java·缓存·mybatis
AI人工智能+电脑小能手11 小时前
【大白话说Java面试题 第65题】【JVM篇】第25题:谈谈对 OOM 的认识
java·开发语言·jvm
阿维的博客日记11 小时前
Nacos 为什么能让配置动态生效?(涉及 @RefreshScope 注解)
java·spring
雨辰AI11 小时前
SpringBoot3 + 人大金仓读写分离 + 分库分表 + 集群高可用 全栈实战
java·数据库·mysql·政务
辰海Coding12 小时前
MiniSpring框架学习-完成的 IoC 容器
java·spring boot·学习·架构
小小编程路13 小时前
C++ 多线程与并发
java·jvm·c++
AI视觉网奇13 小时前
linux 检索库 判断库是否支持
java·linux·服务器