Spring从入门到实战 AOP

AOP

  • AOP:Aspect Oriented Programming面向切面编程

  • 优点

    • 降低模块之间的耦合度
    • 使系统更容易扩展
    • 更好的代码复用
    • 非业务代码更集中,不分散,便于统一管理
    • 业务代码更加简洁纯粹,不掺杂其他代码的影响
  • AOP是对面向对象编程的一个补充,在运行时,动态地将代码切入到类的指定方法、指定位置上的编程思想就是面向切面编程。将不同方法的同一个位置抽象成一个切面对象,对该切面对象进行编程就是AOP。

  • 如何使用

    • 创建Maven工程,pom.xml添加
    	<dependencies>
            <dependency>
                <groupId>org.springframework</groupId>
                <artifactId>spring-context</artifactId>
                <version>5.0.11.RELEASE</version>
            </dependency>
            <dependency>
                <groupId>org.springframework</groupId>
                <artifactId>spring-aop</artifactId>
                <version>5.0.11.RELEASE</version>
            </dependency>
            <dependency>
                <groupId>org.springframework</groupId>
                <artifactId>spring-aspects</artifactId>
                <version>5.0.11.RELEASE</version>
            </dependency>
        </dependencies>
    
    • 创建一个计算器接口,定义4个方法
    package com.southwind.utils;
    
    public interface Cal {
        public int add(int num1,int num2);
        public int sub(int num1,int num2);
        public int mul(int num1,int num2);
        public int div(int num1,int num2);
    }
    
    • 创建接口的实现类CalImpl
    package com.southwind.utils.impl;
    
    import com.southwind.utils.Cal;
    
    public class CakImpl implements Cal {
    
        public int add(int num1, int num2) {
            System.out.println("add方法的参数是[" + num1 + "," + num2 + "]");
            int result = num1+num2;
            System.out.println("add方法的结果是" + result);
            return result;
        }
    
        public int sub(int num1, int num2) {
            System.out.println("sub方法的参数是[" + num1 + "," + num2 + "]");
            int result = num1-num2;
            System.out.println("sub方法的结果是" + result);
            return result;
        }
    
        public int mul(int num1, int num2) {
            System.out.println("mul方法的参数是[" + num1 + "," + num2 + "]");
            int result = num1*num2;
            System.out.println("mul方法的结果是" + result);
            return result;
        }
    
        public int div(int num1, int num2) {
            System.out.println("div方法的参数是[" + num1 + "," + num2 + "]");
            int result = num1/num2;
            System.out.println("div方法的结果是" + result);
            return result;
        }
    }
    
  • 上述代码中,日志信息和业务逻辑的耦合性很高,不利于系统的维护,使用AOP可以进行优化,如何来实现AOP?使用动态代理的方法来实现

    给业务代码找一个代理,打印日志信息的工作交给代理来做,这样的话业务代码就只需要关注自身业务即可

package com.southwind.utils;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.Arrays;

public class MyInyocationHandlen implements InvocationHandler {
    //接受委托对象
    private Object object = null;

    //返回代理对象
    public Object bind(Object object){
        this.object = object;
        return Proxy.newProxyInstance(object.getClass().getClassLoader(),object.getClass().getInterfaces(),this);
    }

    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println(method.getName()+"方法的参数是" + Arrays.toString(args));
        Object result = method.invoke(this.object,args);
        System.out.println(method.getName() + "方法的结果是"+result);
        return result;
    }
}
package com.southwind.test;

import com.southwind.utils.Cal;
import com.southwind.utils.MyInyocationHandlen;
import com.southwind.utils.impl.CakImpl;

public class Test {
    public static void main(String[] args) {
//        Cal cal = new CakImpl();
//        cal.add(1,1);
//        cal.sub(2,1);
//        cal.mul(2,3);
//        cal.div(6,2);

        Cal cal = new CakImpl();
        MyInyocationHandlen myInyocationHandlen = new MyInyocationHandlen();
        Cal proxy = (Cal) myInyocationHandlen.bind(cal);
        proxy.add(1,1);
        proxy.sub(2,1);
        proxy.mul(2,3);
        proxy.div(6,2);
    }
}

以上是通过动态代理实现AOP的过程,比较复杂,不好理解,Spring框架对AOP进行了封装,使用Spring框架可以用面向对象的思想来实现AOP

  • Spring框架中不需要创建InvocationHandler,只需要创建一个切面对象,将所有的非业务代码在切面对象中完成即可,Spring框架底层会自动根据切面类以及目标类生成一个代理对象

LoggerAspect

package com.southwind.aop;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;

import java.util.Arrays;

@Aspect
@Component
public class LoggerAspect {
    @Before("execution(public int com.southwind.utils.impl.CakImpl.*(..))")
    public void before(JoinPoint joinPoint){
        //获取方法名
        String name = joinPoint.getSignature().getName();
        //获取参数
        String args = Arrays.toString(joinPoint.getArgs());
        System.out.println(name+"方法的参数是" + args);
    }

    @After("execution(public int com.southwind.utils.impl.CakImpl.*(..))")
    public void after(JoinPoint joinPoint){
        //获取方法名
        String name = joinPoint.getSignature().getName();
        //获取参数
        String args = Arrays.toString(joinPoint.getArgs());
        System.out.println(name+"方法执行完毕");
    }

    @AfterReturning(value = "execution(public int com.southwind.utils.impl.CakImpl.*(..))",returning = "result")
    public void afterReturning(JoinPoint joinPoint,Object result){
        //获取方法名
        String name = joinPoint.getSignature().getName();
        System.out.println(name+"方法的结果是" + result);
    }

    @AfterThrowing(value = "execution(public int com.southwind.utils.impl.CakImpl.*(..))",throwing = "exception")
    public void afterThrowing(JoinPoint joinPoint,Exception exception){
        //获取方法名
        String name = joinPoint.getSignature().getName();
        System.out.println(name+"方法抛出异常" + exception);
    }

}
  • LoggerAspect类定义处添加两个注解:
    • @Aspect:表示该类是切面类
    • @Component:将该类对象注入到IoC容器
  • 具体方法处添加的注解:
    • @Before:表示方法执行的具体位置和时机
  • CakImpl也需要添加@Component注解,交给IoC来管理
package com.southwind.utils.impl;

import com.southwind.utils.Cal;
import org.springframework.stereotype.Component;

@Component
public class CakImpl implements Cal {

    public int add(int num1, int num2) {
        int result = num1+num2;
        return result;
    }

    public int sub(int num1, int num2) {
        int result = num1-num2;
        return result;
    }

    public int mul(int num1, int num2) {
        int result = num1*num2;
        return result;
    }

    public int div(int num1, int num2) {
        int result = num1/num2;
        return result;
    }
}
  • spring.xml中配置AOP
<?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:contest="http://www.springframework.org/schema/context"
        xmlns:aop="http://www.springframework.org/schema/aop"
        xmlns:p="http://www.springframework.org/schema/p"
        xsi:schemaLocation="
        http://www.springframework.org/schema/aop
        http://www.springframework.org/schema/aop/spring-aop-4.3.xsd
        http://www.springframework.org/schema/beans
	    http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
        http://www.springframework.org/schema/context
        http://www.springframework.org/schema/context/spring-context-4.3.xsd
">

    <!-- 自动扫描 -->
    <contest:component-scan base-package="com.southwind"></contest:component-scan>
    <!-- 使Aspect注解生效,为目标类自动生成代理对象 -->
    <aop:aspectj-autoproxy></aop:aspectj-autoproxy>
</beans>

ontest:component-scan:将com.southwind包中的所有类进行扫描,如果该类同时添加了Component,则将该类扫描到IoC容器中,即IoC管理它的对象。

aop:aspectj-autoproxy:让Spring框架结合切面类和目标类自动生成动态代理对象

  • 切面:横切关注点被模块化的抽象对象
  • 通知:切面对象完成的工作
  • 目标:被通知对象,即被横切的对象
  • 代理:切面、通知、目标混合之后的对象
  • 连接点:通知要插入业务代码的具体位置
  • 切点:AOP通过切点定位到连接点
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值