Spring AOC部分整理

本文详细介绍了Java动态代理机制,通过JDK动态代理实现不修改源代码添加功能,以及AspectJ在AOP操作中的应用。包括切入点表达式、注解配置、通知类型的使用,展示了如何降低模块间的耦合度,实现方法增强。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

面对切面编程

目的:降低各功能模块之间的耦合度

通俗描述:不通过修改源代码方式,在主干功能里面添加新功能

 一、JDK动态代理

jdk动态处理前提是有接口类。创建接口实现类处理对象

1.调用newProxyInstance方法

其中三个参数分别为:类加载器(当前类的加载器,当前类名.class.getClassLoader() )、增强方法所在的类实现的接口(就是前提中已有的接口)、实现这个接口 InvocationHandler(创建代理对象,写增强的部分)

public class JDKProxy {
    public static void main(String[] args) {
        Class[] interfaces = {UserDao1.class};
        UserDaoImpl userDao = new UserDaoImpl();
        UserDao1 dao = (UserDao1) newProxyInstance(JDKProxy.class.getClassLoader(), interfaces, new fangfa(userDao));

UserDao1是现有的接口,UserDaoImpl是接口的实现类,fangfa是写增强部分的类名称

2.创建代理对象代码

invoke方法中是增强部分,即除了执行原有的方法外,还可以在invoke方法内添加想要的功能

invoke三个参数:原有的接口的实现类、要执行的方法(也是原有的实现类中的方法)、要执行的方法的参数。

//创建代理对象
class fangfa implements InvocationHandler {
    //有参构造
    private Object obj;
    public fangfa(Object obj) {
        this.obj = obj;
    }
    //增强部分
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        //方法之前,
        System.out.println("方法执行之前执行~~~" + method.getName() + ":传递的参数~~~" + Arrays.toString(args));
        //被增强的方法执行
        Object res = method.invoke(obj, args);
        //方法之后
        if (method.getName() == "update") {
            System.out.println("haha");
        }
        return res;
    }
}

二、基于 AspectJ 实现 AOP 操作

1、AOP准备工作

1.引入依赖

2.切入点表达式

 切入点表达式作用:知道对哪个类里面的哪个方法进行增强

语法结构: execution([权限修饰符] [返回类型(可不写)] [类全路径] [方法名称]([参数列表]) )

2. AspectJ 注解(重点)

1.在xml文件中,开启注解扫描。增强类和被增强类都要被注解

(注解部分可以参考Spring IOC部分梳理(B)_小曹来巡山的博客-优快云博客

2.在增强类上要在加上Aspect注解(表示生成代理对象),然后在配置文件中配置Aspect,注意要开辟aop名称空间。

<!-- 开启Aspect生成代理对象-->
<aop:aspectj-autoproxy></aop:aspectj-autoproxy>

3.在增强类里面,在作为通知(增强部分)方法上面添加通知类型注解,使用切入点表达式配置

通知类型(Before 前置通知、AfterReturning后置通知、After最终通知、Around环绕通知、AfterThrowing异常通知)

后置通知

 //后置通知(返回通知)
    @AfterReturning(value = "execution(* com.atguigu.spring5.aopanno.User.add(..))")
    public void afterReturning() {
        System.out.println("afterReturning.........");
    }

环绕通知

//环绕通知
    @Around(value = "execution(* com.atguigu.spring5.aopanno.User.add(..))")
    public void around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
        System.out.println("环绕之前.........");

        //被增强的方法执行
        proceedingJoinPoint.proceed();

        System.out.println("环绕之后.........");
    }

引入相同的切入点

在增强类中创建一个空方法,前面加上@Pointcut(value= “execution(~~~)”),后面的增强方法value值为创建的方法名。

 //相同切入点抽取
    @Pointcut(value = "execution(* com.atguigu.spring5.aopanno.User.add(..))")
    public void pointdemo() {

    }

    //前置通知
    //@Before注解表示作为前置通知
    @Before(value = "pointdemo()")
    public void before() {
        System.out.println("before.........");
    }

有多个增强类,设置优先级

在增强类上面添加注解@Order(数字),数字表示优先级

完全使用注解开发

3.AspectJ 配置文件(了解)

配置文件和两个类都没有注解

在xml配置文件中创建两个类对象(增强类和被增强类),(通过两个bean标签创建)

在配置文件中配置文件中切入点

 <!--创建对象-->
    <bean id="book" class="com.atguigu.spring5.aopxml.Book"></bean>
    <bean id="bookProxy" class="com.atguigu.spring5.aopxml.BookProxy"></bean>

    <!--配置aop增强-->
    <aop:config>
        <!--切入点,即需要增强的方法-->
        <aop:pointcut id="p" expression="execution(* com.atguigu.spring5.aopxml.Book.buy(..))"/>
        <!--配置切面-->
        <aop:aspect ref="bookProxy">    <!--bookProxy是增强类-->
            <!--增强作用在具体的方法上-->
            <aop:before method="before" pointcut-ref="p"/>   <!--before是增强方法-->
        </aop:aspect>
    </aop:config>

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值