java spring AOP

AOP

概念
AOP 为 Aspect Oriented Programming 的缩写,意思为面向切面编程,是通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。
AOP 的作用及其优势

  • 作用:在程序运行期间,在不修改源码的情况下对方法进行功能增强
  • 优势:减少重复代码,提高开发效率,并且便于维护

常用的动态代理技术

  • JDK 代理 : 基于接口的动态代理技术
  • cglib 代理: 基于父类的动态代理技术

在这里插入图片描述
JDK代理

/**
 * 接口类
 */
public interface TargetInterface {

    void say();
}

/**
 * 目标类
 */
public class Target implements TargetInterface {
    public void say() {
        System.out.println("目标类执行了");
    }
}


 public static void main( String[] args ) {
        final Target target = new Target();
        TargetInterface proxy = (TargetInterface)Proxy.newProxyInstance(target.getClass().getClassLoader(),
                target.getClass().getInterfaces(),
                new InvocationHandler() {
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                        System.out.println("前面增强");
                        Object invoke = method.invoke(target, args);
                        System.out.println("后置增强");
                        return invoke;
                    }
        });
        proxy.say();
    }
  • Proxy.newProxyInstance() 方法是生成代理对象的,有三个参数,类加载器 ,目标类的所有接口字节码对象,InvocationHandler
  • InvocationHandler : 可以理解为一个回调函数,里面的invoke()是关键,代理对象执行方法就是在执行invoke()方法
  • method.invoke(target, args);属于反射,方法的执行,
public class Target {
    
    public void say1(String name) {
        System.out.println("你好"+name);
    }

    public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InstantiationException, InvocationTargetException {
        Class<Target> targetClass = Target.class;
        Method method= targetClass.getMethod("say1", String.class);
        method.invoke(targetClass.newInstance(),"tom");
    }
}

cglib代理
使用cglib的时候需要导入依赖

<dependency>
  <groupId>org.springframework</groupId>
  <artifactId>spring-context</artifactId>
  <version>5.0.5.RELEASE</version>
</dependency>
<!-- aspectj的织入 -->
<dependency>
  <groupId>org.aspectj</groupId>
  <artifactId>aspectjweaver</artifactId>
  <version>1.8.13</version>
</dependency>

代码

 public static void main( String[] args ) {
    final Target target = new Target(); //创建目标对象
    Enhancer enhancer = new Enhancer();   //创建增强器
    enhancer.setSuperclass(Target.class);//设置父类
    enhancer.setCallback(new MethodInterceptor() { //设置回调
        public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
            System.out.println("前置代码增强....");
            Object invoke = method.invoke(target, objects);
            System.out.println("后置代码增强....");
            return invoke;
        }
    });
    Target proxy =(Target) enhancer.create();//创建代理对象
    proxy.say();
}

AOP 相关概念

Target(目标对象):代理的目标对象

Proxy (代理):一个类被 AOP 织入增强后,就产生一个结果代理类

Joinpoint(连接点):所谓连接点是指那些被拦截到的点。在spring中,这些点指的是方法,因为spring只支持方法类型的连接点。

Pointcut(切入点):所谓切入点是指我们要对哪些 Joinpoint 进行拦截的定义。

Advice(通知/ 增强):所谓通知是指拦截到 Joinpoint 之后所要做的事情就是通知

Aspect(切面):是切入点和通知的结合

Weaving(织入):是指把增强应用到目标对象来创建新的代理对象的过程。spring采用动态代理织入,而AspectJ采用编译期织入和类装载期织入。 安装的过程

书写步骤

  • 1 导入 AOP 相关坐标
  <!--导入spring的context坐标,context依赖aop-->
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-context</artifactId>
      <version>5.0.5.RELEASE</version>
    </dependency>
    <!-- aspectj的织入 -->
    <dependency>
      <groupId>org.aspectj</groupId>
      <artifactId>aspectjweaver</artifactId>
      <version>1.8.13</version>
    </dependency>

  • 2 创建目标接口和目标类(内部有切点)
/**
 * 接口类
 */
public interface TargetInterface {

    void say();
}

/**
 * 目标类
 */
public class Target implements TargetInterface {
 	@Override
    public void say() {
        try {
            Thread.sleep(2000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("目标类执行了");
    }
}

  • 3 创建切面类(内部有增强方法)
public class MyAspect {
     //前置增强方法
    public void before(){
        System.out.println("前置代码增强了");
    }
    //后置增强方法
    public void after(){
        System.out.println("后置");
    }
    //环绕增强
    public Object around(ProceedingJoinPoint p){
        long start = System.currentTimeMillis();
        String className = p.getSignature().getDeclaringTypeName();
        String method =  p.getSignature().getName();
        Object proceed = null;
        try {
            proceed = p.proceed();
        } catch (Throwable throwable) {
            throwable.printStackTrace();
        }
        long end = System.currentTimeMillis();
        System.out.println("类名:" + className);
        System.out.println("方法名是:" + method);
        System.out.println("执行时间长是:"+(end-start));
        return proceed;
    }
}
  • 4 将目标类和切面类的对象创建权交给 spring
<!--配置目标类-->
<bean id="target" class="com.xz.service.impl.Target"/>
<!--配置切面类-->
<bean id="myAspect" class="com.xz.aspect.MyAspect"/>
  • 5 在 applicationContext.xml 中配置织入关系
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="
       http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/aop
       http://www.springframework.org/schema/aop/spring-aop.xsd">

    <bean id="myAspect" class="com.xinzhi.aspect.MyAspect" />

    <bean id="target" class="com.xinzhi.target.impl.TargetInterfaceImpl" />

    <aop:config>
        <aop:aspect ref="myAspect">
<!--            <aop:before method="before" pointcut="execution(* com.xinzhi.target.impl.*.*(..))"/>-->
<!--            <aop:after method="after" pointcut="execution(* com.xinzhi.target.impl.*.*(..))"/>-->
            <aop:around method="around" pointcut="execution(* com.xz.target.impl.*.*(..))"/>
        </aop:aspect>
    </aop:config>

</beans>
  • 6 测试代码
public static void main( String[] args ) {
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext("application.xml");
        TargetInterface bean = (TargetInterface)applicationContext.getBean("target");
        bean.say();
 }

2 切点表达式

execution([修饰符] 返回值类型 包名.类名.方法名(参数))
	常用的书写格式:
	execution(* 包名.*.*(..))
  • 访问修饰符可以省略
  • 返回值类型、包名、类名、方法名可以使用星号* 代表任意
  • 包名与类名之间一个点 . 代表当前包下的类,两个点 … 表示当前包及其子包下的类
  • 参数列表可以使用两个点 … 表示任意个数,任意类型的参数列表

注解的AOP

1 书写步骤

  • 1创建目标接口和目标类(内部有切点)
  • 2 创建切面类(内部有增强方法)
  • 3 将目标类和切面类的对象创建权交给 spring
  • 4 在切面类中使用注解配置织入关系
  • 5 在配置文件中开启组件扫描和 AOP 的自动代理
  • 6 测试
/**
 * 目标类
 */
@Component("target")
public class Target implements TargetInterface {

    public void say() {
        System.out.println("目标类执行了");
    }
}

@Component
@Aspect
public class MyAspect {
    //前置增强方法
    @Before("execution(* com.xz.target.impl.*.*(..))")
    public void before(){
        System.out.println("前置代码增强了");
    }
    //后置增强方法
    @After("execution(* com.xz.target.impl.*.*(..))")
    public void after(){
        System.out.println("后置");
    }
    //环绕增强
    @Around("execution(* com.xz.target.impl.*.*(..))")
    public Object around(ProceedingJoinPoint p){
        long start = System.currentTimeMillis();
        String className = p.getSignature().getDeclaringTypeName();
        String method =  p.getSignature().getName();
        Object proceed = null;
        try {
            proceed = p.proceed();
        } catch (Throwable throwable) {
            throwable.printStackTrace();
        }
        long end = System.currentTimeMillis();
        System.out.println("类名:" + className);
        System.out.println("方法名是:" + method);
        System.out.println("执行时间长是:"+(end-start));
        return proceed;
    }
}

<!--组件扫描-->
<context:component-scan base-package="com.xz"/>
<!--aop的自动代理-->
<aop:aspectj-autoproxy></aop:aspectj-autoproxy>
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java Spring AOP (Aspect-Oriented Programming) 是 Spring 框架中的一个重要特性。它允许你通过将横切关注点(cross-cutting concerns)与程序的核心业务逻辑分离开来,以一种模块化的方式来管理和处理它们。 AOP 提供了一种在程序运行时动态地将通用功能(例如日志记录、性能计数、事务管理等)应用到应用程序中各个模块的方法上的方法。通过使用 Spring AOP,你可以将这些通用功能从核心业务逻辑中解耦出来,以便更好地维护和理解代码。 在 Spring AOP 中,你可以通过使用切点(pointcut)和通知(advice)来定义横切关注点。切点指定了在应用程序中哪些方法需要被拦截,而通知则定义了在拦截点(join point)上执行的操作。 Spring AOP 支持多种类型的通知,包括前置通知(Before advice)、后置通知(After advice)、返回通知(After-returning advice)、异常通知(After-throwing advice)和环绕通知(Around advice)。这些通知可以在方法执行前、后或周围被触发,以便执行与横切关注点相关的行为。 通过配置 Spring AOP,你可以将切点和通知与目标对象关联起来,并在运行时自动应用这些通知。这样,你就能够以一种声明式的方式管理和组织应用程序中的横切关注点,而无需修改核心业务逻辑。 总括来说,Java Spring AOPSpring 框架中的一个功能强大的特性,它通过将横切关注点与核心业务逻辑解耦,提供了一种模块化和可维护的方式来处理通用功能。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值