静态代理和动态代理

  • 静态代理

    • 什么是静态代理

      • 由程序创建或特定工具自动生成源代码,在程序运行前,代理类的.class文件就已经存在
      • 通过将目标类与代理类实现同一个接口,让代理类持有真实类对象,然后在代理类方法中调用真实类方法,在调用真实类方法的前后添加我们所需要的功能扩展代码来达到增强的目的
      • A -> B -> C
    • 优点

      • 代理使客户端不需要知道实现类是什么,怎么做的,而客户端只需知道代理即可
      • 方便增加功能,拓展业务逻辑
    • 缺点

      • 代理类中出现大量冗余的代码,非常不利于扩展和维护
      • 如果接口增加一个方法,除了所有实现类需要实现这个方法外,所有代理类也需要实现此方法。增加了代码维护的复杂度
  • 动态代理

    • 在程序运行时,运用反射机制动态创建而成,无需手动编写代码
    • SpringAOP里面常用的是 JDK动态代理、CGLIB动态代理
    • JDK动态代理,要求目标对象实现一个接口,但是有时候目标对象只是一个单独的对象,并没有实现任何的接口,这个时候就可以用CGLib动态代理
    • JDK动态代理是自带的,CGlib需要引入第三方包
    • CGLib动态代理,它是在内存中构建一个子类对象从而实现对目标对象功能的扩展
    • CGLib动态代理基于继承来实现代理,所以无法对final类、private方法和static方法实现代理
  • Spring AOP中的代理使用的默认策略?

    • 如果目标对象实现了接口,则默认采用JDK动态代理
    • 如果目标对象没有实现接口,则采用CgLib进行动态代理
    • 如果目标对象实现了接口,程序里面依旧可以指定使用CGlib动态代理
### 静态代理动态代理的概念、区别及使用场景 #### 1. 概念 静态代理动态代理代理模式的两种实现方式。代理模式的核心思想是通过一个代理对象来控制对目标对象的访问,从而实现功能扩展或访问控制。 - **静态代理**是指在编译时就已经确定了代理代理类的关系[^1]。代理类需要手动编写,并且必须实现与被代理类相同的接口。 - **动态代理**则是在运行时动态生成代理类[^2],无需在编译前手动编写代理类代码。动态代理通常通过 Java 的 `java.lang.reflect.Proxy` 类或其他框架(如 CGLIB)实现。 #### 2. 区别 静态代理动态代理的主要区别体现在以下几个方面: - **定义时间**: - 静态代理代理类在编译时就已经确定[^3]。 - 动态代理代理类是在运行时动态生成的[^1]。 - **灵活性**: - 静态代理由于代理类是固定的,因此不够灵活,每次新增接口或修改接口时都需要重新编写代理类。 - 动态代理可以适应多种接口的变化,具有更高的灵活性可扩展性[^3]。 - **性能**: - 静态代理因为代理类是提前编译好的,所以在执行效率上可能略高于动态代理。 - 动态代理由于需要在运行时生成代理类,可能会引入一定的性能开销,但现代 JVM 已经对此进行了优化[^1]。 - **适用场景**: - 静态代理适合于接口较少且相对固定的情况。 - 动态代理更适合于接口较多或经常变化的场景,能够减少代码冗余并提高开发效率[^3]。 #### 3. 使用场景 - **静态代理**: - 当代理类的功能较为简单,且接口不常变化时,可以使用静态代理[^1]。 - 示例:日志记录、权限校验等简单的功能增强[^2]。 - **动态代理**: - 在需要为多个类提供统一的代理逻辑时,动态代理更加合适[^3]。 - 示例:AOP(面向切面编程)、事务管理、远程方法调用等复杂场景。 #### 示例代码 以下分别展示了静态代理动态代理实现方式。 ##### 静态代理示例 ```java // 定义接口 interface Rent { void rent(); } // 真实主题类 class RealRent implements Rent { @Override public void rent() { System.out.println("房东出租房子"); } } // 静态代理类 class ProxyRent implements Rent { private Rent realRent; public ProxyRent(Rent realRent) { this.realRent = realRent; } @Override public void rent() { System.out.println("中介开始找房..."); realRent.rent(); System.out.println("中介收取佣金"); } } // 测试 public class StaticProxyTest { public static void main(String[] args) { Rent realRent = new RealRent(); Rent proxyRent = new ProxyRent(realRent); proxyRent.rent(); // 输出:中介开始找房... 房东出租房子 中介收取佣金 } } ``` ##### 动态代理示例 ```java import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; // 定义接口 interface Rent { void rent(); } // 真实主题类 class RealRent implements Rent { @Override public void rent() { System.out.println("房东出租房子"); } } // 动态代理类 class DynamicProxyHandler implements InvocationHandler { private Object target; public DynamicProxyHandler(Object target) { this.target = target; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("中介开始找房..."); Object result = method.invoke(target, args); System.out.println("中介收取佣金"); return result; } } // 测试 public class DynamicProxyTest { public static void main(String[] args) { Rent realRent = new RealRent(); InvocationHandler handler = new DynamicProxyHandler(realRent); Rent proxyRent = (Rent) Proxy.newProxyInstance( realRent.getClass().getClassLoader(), realRent.getClass().getInterfaces(), handler ); proxyRent.rent(); // 输出:中介开始找房... 房东出租房子 中介收取佣金 } } ```
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值