浅谈动态代理

什么是动态代理?


以下为个人理解:

  • 动态代理就是在程序运行的期间,动态地针对对象的方法进行增强操作。并且这个动作的执行者已经不是"this"对象了,而是我们创建的代理对象,这个代理对象就是类似中间人的角色,帮助我们为目标方法嵌入一些其他的逻辑进去。

jdk动态代理的原理


  • jdk自带的动态代理的工作原理是利用反射的newInstance,创建一个代理对象(proxy),获取到目标接口的方法,然后我们就可以在Invoke之前或之后做操作。它抽取出了一个invokeHandler,里面就有目标method。

  • 试想一下,当我们拥有了class对象,增强逻辑(invokeHandler),也就是增强的目标方法之后,我们自己利用反射(不利用Proxy)也可以很容易写出我们自己的动态代理。但是问题就在于,我们只有明确了目标类之后,通过自己编写Proxy类,实现目标接口,往里面塞invoktionHandler, 最后New出来这个实实在在的对象。而jdk提供的动态代理,却可以在并不知情的情况下,帮我们做这一系列的动作。

  • jdk生成代理类时,并没有经历源码阶段,编译阶段,而是直接到字节码阶段,它生成的代理类是看不到的,因为它直接就是字节码文件了。 它帮我们继承了Proxy类,并且还动态的帮助我们继承了目标接口。也就是说,它帮我们写代码了。里面用到的技术,是ASM技术,它可以直接生成我们想要的字节码。

  • jdk方法反射调用优化:

  • invoke() 利用反射进行本地调用,效率低下,它在调用了一定的次数(16)之后会生成实例化对象,变成正常调用。

InvocationHandler接口

publicinterfaceInvocationHandler {
​
    publicObjectinvoke(Objectproxy, Methodmethod, Object[] args)throwsThrowable;
        
}

源码

@CallerSensitive
publicstaticObjectnewProxyInstance(ClassLoaderloader,
                                      Class<?>[] interfaces,
                                      InvocationHandlerh)
    throwsIllegalArgumentException
{
    Objects.requireNonNull(h);
​
    finalClass<?>[] intfs=interfaces.clone();
    finalSecurityManagersm=System.getSecurityManager();
    if (sm!=null) {
        checkProxyAccess(Reflection.getCallerClass(), loader, intfs);
    }
​
    /*
     * Look up or generate the designated proxy class.
     */
    Class<?>cl=getProxyClass0(loader, intfs);
​
    /*
     * Invoke its constructor with the designated invocation handler.
     */
    try {
        if (sm!=null) {
            checkNewProxyPermission(Reflection.getCallerClass(), cl);
        }
​
        finalConstructor<?>cons=cl.getConstructor(constructorParams);
        finalInvocationHandlerih=h;
        if (!Modifier.isPublic(cl.getModifiers())) {
            AccessController.doPrivileged(newPrivilegedAction<Void>() {
                publicVoidrun() {
                    cons.setAccessible(true);
                    returnnull;
                }
            });
        }
        returncons.newInstance(newObject[]{h});
    } catch (IllegalAccessException|InstantiationExceptione) {
        thrownewInternalError(e.toString(), e);
    } catch (InvocationTargetExceptione) {
        Throwablet=e.getCause();
        if (tinstanceofRuntimeException) {
            throw (RuntimeException) t;
        } else {
            thrownewInternalError(t.toString(), t);
        }
    } catch (NoSuchMethodExceptione) {
        thrownewInternalError(e.toString(), e);
    }
}

cglib动态代理的原理


  • cglib动态代理的原理跟jdk的类似,只不过它是基于父类继承,也就是不需要实现接口就可以做增强。它的内部并不是InvoktionHandler,而是方法拦截器 MethodInterceptor 。

  • 前面的jdk动态代理,它是利用ASM技术帮我们动态编写了一个proxy对象,其中继承了Proxy父类,实现了目标接口,而cglib则是利用ASM直接帮助我们继承了目标类,不需要接口。

  • 并且有所区别的是,它不仅仅通过反射拿到method,还拿到了MethodProxy。

  • MethodInterceptor接口 继承了Callback接口,它的拦截方法里面有一个特殊的参数 MethodProxy,这玩意可以不通过反射调用方法,通过invokeSuper() 方法可以直接正常调用。

  • MethodProxy是怎么做到正常调用的?

  • 其实就是我们前面提到的,继承了Proxy父类之后,就得到了父类的原始方法,当我调用invokeSuper的时候,直接调用的就是父类的原始方法。

MethodInterceptor

publicinterfaceMethodInterceptorextendsCallback {
    Objectintercept(Objectvar1, Methodvar2, Object[] var3, MethodProxyvar4) throwsThrowable;
}

MethodProxy

  • invoke() 无反射调用

  • invokeSuper() 无反射调用

publicObjectinvoke(Objectobj, Object[] args) throwsThrowable {
    try {
        this.init();
        MethodProxy.FastClassInfofci=this.fastClassInfo;
        returnfci.f1.invoke(fci.i1, obj, args);
    } catch (InvocationTargetExceptionvar4) {
        throwvar4.getTargetException();
    } catch (IllegalArgumentExceptionvar5) {
        if (this.fastClassInfo.i1<0) {
            thrownewIllegalArgumentException("Protected method: "+this.sig1);
        } else {
            throwvar5;
        }
    }
}
publicObjectinvokeSuper(Objectobj, Object[] args) throwsThrowable {
    try {
        this.init();
        MethodProxy.FastClassInfofci=this.fastClassInfo;
        returnfci.f2.invoke(fci.i2, obj, args);
    } catch (InvocationTargetExceptionvar4) {
        throwvar4.getTargetException();
    }
}
<think>好的,我现在需要回答用户关于IoC(控制反转)和AOP(面向切面编程)的基本概念、原理与区别的问题。用户已经提供了站内引用中的相关信息,我需要结合这些信息以及之前的对话历史来构建一个结构清晰的回答。 首先,我需要回顾之前的对话。用户之前询问了数据库索引的相关问题,我回答了定义、优缺点,并生成了相关问题。现在用户转向了Spring框架中的IoC和AOP,这属于Java企业级开发的核心概念。 接下来,我需要解析用户的问题。用户明确要求了解IoC和AOP的概念、原理以及两者的区别。根据提供的引用,IoC涉及控制反转和依赖注入,而AOP则涉及面向切面编程,用于处理交叉关注点。同时,引用中还提到了Spring框架如何结合这两个概念来降低耦合度,提高代码复用性。 我需要先分别解释IoC和AOP的基本概念,确保每个部分都有明确的定义。例如,IoC的核心是将对象的创建和依赖管理交给容器,而不是由程序自己控制。而AOP则是通过动态代理技术,将横切关注点(如日志、事务)模块化。 然后,要深入原理部分。对于IoC,可能需要解释依赖注入的几种方式(构造器注入、setter注入等),以及Spring容器如何管理Bean的生命周期。对于AOP,需要说明动态代理的实现方式(JDK动态代理和CGLIB),以及如何通过切入点表达式定义切点,通知的类型(前置、后置、环绕等)及其执行流程。 接下来是区别部分。需要比较两者的关注点不同:IoC主要解决对象创建和依赖管理,而AOP处理分散在多个模块中的横切关注点。同时,它们在实现机制上的不同,IoC通过容器管理Bean,AOP通过代理和拦截器实现。 此外,用户提供的引用中提到AOP在BeanPostProcessor后置处理中实现,这一点需要在原理部分详细说明。同时,结合应用场景,如事务管理和日志记录,可以具体举例说明两者的实际用途。 需要确保回答结构清晰,分点列出,每个部分都有小标题,并使用适当的格式如加粗、列表等增强可读性。同时,注意引用来源,比如在解释IoC和AOP时引用提供的文献[^1][^2][^3][^4]。 最后,生成相关问题,考虑用户可能想深入了解的内容,比如IoC容器的工作机制、AOP的实现技术比较,或两者在项目中的实际应用案例。 需要检查是否符合之前的格式要求:使用中文,行内数学表达式用$...$,独立公式用$$...$$(虽然这个问题可能不需要数学公式),但可能需要代码块示例,比如动态代理的简单代码。但用户提供的引用中没有代码示例,可能不需要添加,除非有必要。 另外,注意用户提到的“参考站内引用”,但回答中需要自然地融入这些引用,而不是直接复制。需要用自己的话组织内容,并在适当的位置标注引用标识,如[^1][^2]等。 总结步骤: 1. 定义IoC和AOP,分别说明概念。 2. 分析两者的实现原理,包括技术细节(如依赖注入方式、动态代理)。 3. 比较两者的区别,从目的、实现方式、应用场景等方面。 4. 结合实际应用举例,如事务管理、日志记录。 5. 生成相关问题,引导用户深入探讨。 现在需要确保每个部分的信息准确,引用正确,并且语言流畅,结构清晰。</think>### IoC(控制反转)与AOP(面向切面编程)解析 --- #### 一、IoC(控制反转) **1. 基本概念** IoC(Inversion of Control)是一种**设计思想**,核心是将对象的创建与依赖关系的控制权从程序代码转移给外部容器(如Spring容器)。传统编程中,对象直接通过`new`关键字创建并管理依赖,而IoC通过**依赖注入(DI)**或依赖查找实现控制权的反转。例如: ```java // 传统方式:对象在代码中直接创建 UserService userService = new UserServiceImpl(); // IoC方式:依赖由容器注入 @Autowired private UserService userService; ``` **2. 实现原理** - **依赖注入(DI)**:Spring容器通过构造器、Setter方法或字段反射注入依赖对象。 - **Bean生命周期管理**:容器负责Bean的实例化、初始化、使用和销毁,通过配置文件或注解(如`@Component`)定义Bean。 - **解耦优势**:对象间依赖关系由容器配置,降低代码耦合度,提升可测试性和维护性[^3]。 **3. 应用场景** - 模块化开发:不同模块通过接口依赖,实现灵活替换。 - 单元测试:Mock对象可轻松注入,无需硬编码依赖[^3]。 --- #### 二、AOP(面向切面编程) **1. 基本概念** AOP(Aspect-Oriented Programming)通过**动态代理技术**,将分散在多个模块中的**横切关注点**(如日志、事务)抽象为独立的切面,实现功能的无
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值