spring5动态代理AOP

代理模式 为什么要学习代理模式?
因为这就是SpringAOP的底层

我们的业务逻辑是这样的
dao层——>service层——>controller层——>前端

动态代理和静态代理一样,动态代理是动态生成的。
在了解动态代理之前先了解两个类
Proxy:代理类,它里面有个方法很重要newProxyInstance()
InvokationHandler:接口它里面有个抽象方法 就是对方法进行增强的具体操作

AOP:在不改变项目源码的情况下,实现对某一方法的改动 就是AOP面向切面编程,AOP的底层使用的是动态代理

动态代理操作
首先,确定要操作的接口方法 与接口实现类

接口`


public interface Show {

        void gethost();

}

接口实现类

public class Host implements Show{
    @Override
    public void gethost() {
        System.out.println("房子出租");
    }
}

动态代理类

package com.mumu.dao01;


import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
public class  Proxy1 implements InvocationHandler{

    public static void main(String[] args) {

        /**  将接口传递过来 newProxyInstance要用   */
       Class[] interfaces={Show.class};

        /**  用 Proxy类 调用 newProxyInstance方法 参数1;当前所在类的加载器,   参数2;需要切入的接口   参数3;代理类的对象 */
        Show o = (Show)Proxy.newProxyInstance(Proxy1.class.getClassLoader(), interfaces, new Proxy1(new Host()));
        o.gethost(); //调用接口的方法
    }

    //将接口的实现类传递过来
    Object obj;

    //用构造器的方式传递过来
    public Proxy1(Object obj){
        this.obj=obj;
    }




    /**  这个接口的实现类方法就是对增强接口方法的 具体操作          */
    @Override   //参数1;接口实现类的对象  参数2;代理的方法  参数3;数组
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

        System.out.println("12");

        Object invoke = method.invoke(obj, args);// 这个调用就是调用接口中 原始的方法   在它上面或下面 都可以写代码  增强方法
        System.out.println("45");
        return invoke;
    }
}



了解完动态代理
我们来了解AOP操作术语
连接点:类里面哪些地方可以被增强,这些方法就叫连接点
切入点:实际真正增强的方法,成为切入点
通知(增强):实际增强的逻辑部分称为通知
切面:是动作

AOP操作准备:
1 spring框架一般都是基于AspectJ实现AOP操作
什么是AspectJ?
AspevcJ是独立的框架,它不是spring的组成部分,一般把AspevcJ一起使用,进行AOP操作

首先了解切入点表达式:

表示要对com.mumu.test.Userimp.add()方法进行切面操作
execution(* com.mumu.test.Userimp.add(..))

它配合注解使用
例如

        //声明在哪个方法切入     @Before 是在前置切入
    @Before(value = "execution(* com.mumu.test.Userimp.add(..))")

切入注的相关解析

需要增强方法

package com.mumu.edu.test;

import org.springframework.stereotype.Component;

@Component
public class Userimp {

    public void myince() {

        System.out.println("我是userimp");
    }
}

代理类

@Component
@Aspect//加上这个注解  表示这是个切面声明
public class Porxy02 {


        //声明在哪个方法切入     @Before 是在前置切入
    @Before(value = "execution(* com.mumu.test.Userimp.add(..))")
    public void  before(){

        System.out.println("前置");

    }

}

配置文件

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
                            http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd
                            http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd">
    <!--        开启包扫描-->
    <context:component-scan base-package="com.mumu.test">   </context:component-scan>



<!--        开启aop-->
    <aop:aspectj-autoproxy></aop:aspectj-autoproxy>
</beans>



测试类

package com.mumu.test;


import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class test {
    public static void main(String[] args) {

        ApplicationContext context = new ClassPathXmlApplicationContext("bean.xml");
        Userimp userAop = context.getBean("userimp", Userimp.class);
        userAop.add();

    }


}

注意点 我们在使用AspectJ时 要导包

完成操作后
那么 问题来了,如果有多个方法 都想对某一个方法进行增强操作
如何判定优先级?
只需要在增强类上面添加注解@Order(数值型值) 数字值越小 优先级越高

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring AOP中,动态代理是实现AOP的一种方式。Spring AOP使用了Java的动态代理机制来创建代理对象,并将切面逻辑织入到目标对象的方法调用中。 动态代理是在运行时生成代理对象的一种机制。它不需要在编写代码时就确定要代理的类和方法,而是在运行时根据需要创建代理对象。在Spring AOP中,主要有两种类型的动态代理:基于接口的代理和基于类的代理。 1. 基于接口的代理(JDK动态代理):当目标对象实现了至少一个接口时,Spring AOP会使用JDK动态代理生成代理对象。JDK动态代理通过实现目标对象所实现的接口来生成代理对象,在调用代理对象的方法时,会通过InvocationHandler接口将方法调用转发给实际的目标对象。 2. 基于类的代理(CGLIB动态代理):当目标对象没有实现任何接口时,Spring AOP会使用CGLIB动态代理生成代理对象。CGLIB动态代理通过继承目标对象生成一个子类,并覆盖其中的方法来实现代理。调用代理对象的方法时,会先进入子类的方法,然后再调用目标对象的方法。 使用动态代理可以实现对目标对象的拦截和增强,而不需要修改目标对象的源代码。代理对象可以在目标对象的方法执行前、执行后或执行过程中插入额外的逻辑,例如日志记录、性能统计、事务管理等。 需要注意的是,动态代理只能对公共方法进行拦截,对私有方法、静态方法或final方法无法进行拦截。同时,动态代理也只能拦截通过代理对象调用的方法,直接通过目标对象调用方法时无法实现拦截和增强。 总结来说,Spring AOP使用动态代理来实现切面逻辑的织入,可以通过JDK动态代理或CGLIB动态代理生成代理对象,并将切面逻辑应用到目标对象的方法调用中。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值