Spring AOP详细介绍

AOP称为面向切面编程,在程序开发中主要用来解决一些系统层面上的问题,比如日志,事务,权限等待,Struts2的拦截器设计就是基于AOP的思想,是个比较经典的例子。

一 AOP的基本概念

(1)Aspect(切面):通常是一个类,里面可以定义切入点和通知

(2)JointPoint(连接点):程序执行过程中明确的点,一般是方法的调用

(3)Advice(通知):AOP在特定的切入点上执行的增强处理,有before,after,afterReturning,afterThrowing,around

(4)Pointcut(切入点):就是带有通知的连接点,在程序中主要体现为书写切入点表达式

(5)AOP代理:AOP框架创建的对象,代理就是目标对象的加强。Spring中的AOP代理可以使JDK动态代理,也可以是CGLIB代理,前者基于接口,后者基于子类

二 Spring AOP

Spring中的AOP代理还是离不开Spring的IOC容器,代理的生成,管理及其依赖关系都是由IOC容器负责,Spring默认使用JDK动态代理,在需要代理类而不是代理接口的时候,Spring会自动切换为使用CGLIB代理,不过现在的项目都是面向接口编程,所以JDK动态代理相对来说用的还是多一些。

三 基于注解的AOP配置方式

1.启用@AsjectJ支持

在applicationContext.xml中配置下面一句:

<aop:aspectj-autoproxy />
2.通知类型介绍

(1)Before:在目标方法被调用之前做增强处理,@Before只需要指定切入点表达式即可

(2)AfterReturning:在目标方法正常完成后做增强,@AfterReturning除了指定切入点表达式后,还可以指定一个返回值形参名returning,代表目标方法的返回值

(3)AfterThrowing:主要用来处理程序中未处理的异常,@AfterThrowing除了指定切入点表达式后,还可以指定一个throwing的返回值形参名,可以通过该形参名

来访问目标方法中所抛出的异常对象

(4)After:在目标方法完成之后做增强,无论目标方法时候成功完成。@After可以指定一个切入点表达式

(5)Around:环绕通知,在目标方法完成前后做增强处理,环绕通知是最重要的通知类型,像事务,日志等都是环绕通知,注意编程中核心是一个ProceedingJoinPoint

3.例子:

(1)Operator.java --> 切面类

@Component
@Aspect
public class Operator {
    
    @Pointcut("execution(* com.aijava.springcode.service..*.*(..))")
    public void pointCut(){}
    
    @Before("pointCut()")
    public void doBefore(JoinPoint joinPoint){
        System.out.println("AOP Before Advice...");
    }
    
    @After("pointCut()")
    public void doAfter(JoinPoint joinPoint){
        System.out.println("AOP After Advice...");
    }
    
    @AfterReturning(pointcut="pointCut()",returning="returnVal")
    public void afterReturn(JoinPoint joinPoint,Object returnVal){
        System.out.println("AOP AfterReturning Advice:" + returnVal);
    }
    
    @AfterThrowing(pointcut="pointCut()",throwing="error")
    public void afterThrowing(JoinPoint joinPoint,Throwable error){
        System.out.println("AOP AfterThrowing Advice..." + error);
        System.out.println("AfterThrowing...");
    }
    
    @Around("pointCut()")
    public void around(ProceedingJoinPoint pjp){
        System.out.println("AOP Aronud before...");
        try {
            pjp.proceed();
        } catch (Throwable e) {
            e.printStackTrace();
        }
        System.out.println("AOP Aronud after...");
    }
    
}

(2)UserService.java --> 定义一些目标方法

@Service
public class UserService {
    
    public void add(){
        System.out.println("UserService add()");
    }
    
    public boolean delete(){
        System.out.println("UserService delete()");
        return true;
    }
    
    public void edit(){
        System.out.println("UserService edit()");
        int i = 5/0;
    }
    
    
}

(3).applicationContext.xml

<context:component-scan base-package="com.aijava.springcode"/>
    
<aop:aspectj-autoproxy />

 

(4).Test.java 

 

public class Test {
    public static void main(String[] args) {
        ApplicationContext ctx = new ClassPathXmlApplicationContext("classpath:applicationContext.xml");
        UserService userService = (UserService) ctx.getBean("userService");
        userService.add();
    }
}

上面是一个比较简单的测试,基本涵盖了各种增强定义。注意:做环绕通知的时候,调用ProceedingJoinPoint的proceed()方法才会执行目标方法。

4.通知执行的优先级

进入目标方法时,先织入Around,再织入Before,退出目标方法时,先织入Around,再织入AfterReturning,最后才织入After。

注意:Spring AOP的环绕通知会影响到AfterThrowing通知的运行,不要同时使用!同时使用也没啥意义。

5.切入点的定义和表达式

切入点表达式的定义算是整个AOP中的核心,有一套自己的规范

Spring AOP支持的切入点指示符:

(1)execution:用来匹配执行方法的连接点

A:@Pointcut("execution(* com.aijava.springcode.service..*.*(..))")

第一个*表示匹配任意的方法返回值,..(两个点)表示零个或多个,上面的第一个..表示service包及其子包,第二个*表示所有类,第三个*表示所有方法,第二个..表示

方法的任意参数个数

B:@Pointcut("within(com.aijava.springcode.service.*)")

within限定匹配方法的连接点,上面的就是表示匹配service包下的任意连接点

C:@Pointcut("this(com.aijava.springcode.service.UserService)")

this用来限定AOP代理必须是指定类型的实例,如上,指定了一个特定的实例,就是UserService

D:@Pointcut("bean(userService)")

bean也是非常常用的,bean可以指定IOC容器中的bean的名称

6.基于XML形式的配置方式

开发中如果选用XML配置方式,通常就是POJO+XML来开发AOP,大同小异,无非就是在XML文件中写切入点表达式和通知类型

例子:

(1)Log.java

public class Log {

    private Integer id;

    //操作名称,方法名
    private String operName;

    //操作人
    private String operator;

    //操作参数
    private String operParams;

    //操作结果 成功/失败
    private String operResult;

    //结果消息
    private String resultMsg;

    //操作时间
    private Date operTime = new Date();

    setter,getter

}

(2).Logger.java

/**
 * 日志记录器 (AOP日志通知)
 */
public class Logger {
    
    @Resource
    private LogService logService;
    
    public Object record(ProceedingJoinPoint pjp){
        
        Log log = new Log();
        try {
            log.setOperator("admin");
            String mname = pjp.getSignature().getName();
            log.setOperName(mname);
            
            //方法参数,本例中是User user
            Object[] args = pjp.getArgs();
            log.setOperParams(Arrays.toString(args));
            
            //执行目标方法,返回的是目标方法的返回值,本例中 void
            Object obj = pjp.proceed();
            if(obj != null){
                log.setResultMsg(obj.toString());
            }else{
                log.setResultMsg(null);
            }
            
            log.setOperResult("success");
            log.setOperTime(new Date());
            
            return obj;
        } catch (Throwable e) {
            log.setOperResult("failure");
            log.setResultMsg(e.getMessage());
        } finally{
            logService.saveLog(log);
        }
        return null;
    }
}

(3).applicationContext.xml

<aop:config>
        <aop:aspect id="loggerAspect" ref="logger">
            <aop:around method="record" pointcut="(execution(* com.aijava.distributed.ssh.service..*.add*(..))
                                              or   execution(* com.aijava.distributed.ssh.service..*.update*(..))
                                              or   execution(* com.aijava.distributed.ssh.service..*.delete*(..)))
                                            and !bean(logService)"/>
        </aop:aspect>
</aop:config>

 

注意切入点表达式,!bean(logService) 做日志通知的时候,不要给日志本身做日志,否则会造成无限循环!

有关更详细的Spring AOP知识,可以查看Spring官方文档第9章Aspect Oriented Programming with Spring 

7.JDK动态代理介绍

例子:

(1)UserService.java

public interface UserService {
    
    public void add();
}

 (2)UserServiceImpl.java

public class UserServiceImpl implements UserService{

    public void add() {
        System.out.println("User add()...");
    }
    
}

(3)ProxyUtils.java

public class ProxyUtils implements InvocationHandler{
    
    private Object target;
    
    public ProxyUtils(Object target){
        this.target = target;
    }
    
    public Object getTarget() {
        return target;
    }
    
    public void setTarget(Object target) {
        this.target = target;
    }

    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("do sth before...");
        method.invoke(target, args);
        System.out.println("do sth after...");
        return null;
    }

}

(4)Test.java

public class Test {
    public static void main(String[] args) {
        UserService userService = new UserServiceImpl();
        ProxyUtils proxyUtils = new ProxyUtils(userService);
        UserService proxyObject = (UserService) Proxy.newProxyInstance(Thread.currentThread().getContextClassLoader(),UserServiceImpl.class.getInterfaces(), proxyUtils);
        proxyObject.add();
    }
}

 

1 public interface IHello {
2     /**
3      * 业务方法
4      * @param str
5      */
6     void sayHello(String str);
7 }

  目标类代码:

1 public class Hello implements IHello{
2     @Override
3     public void sayHello(String str) {
4         System.out.println("hello "+str);
5     }
6     
7 }

  代理类代码,我们给它添加日志记录功能,在方法开始前后执行特定的方法,是不是和AOP特别像呢?

public class ProxyHello implements IHello{    
    private IHello hello;    
    public ProxyHello(IHello hello) {
        super();
        this.hello = hello;
    }
    @Override
    public void sayHello(String str) {
        Logger.start();//添加特定的方法
        hello.sayHello(str);
        Logger.end();
    }

}

  日志类代码:

1 public class Logger {
2     public static void start(){
3         System.out.println(new Date()+ " say hello start...");
4     }
5     
6     public static void end(){
7         System.out.println(new Date()+ " say hello end");
8     }
9 }

  测试代码:

1 public class Test {
2     public static void main(String[] args) {
3         IHello hello = new ProxyHello(new Hello());//如果我们需要日志功能,则使用代理类
4         //IHello hello = new Hello();//如果我们不需要日志功能则使用目标类
5         hello.sayHello("明天");    
6     }
7 }

这样我们就实现了一个最简单的AOP,但是这样会存在一个问题:如果我们像Hello这样的类很多,那么,我们是不是要去写很多个HelloProxy这样的类呢。其实也是一种很麻烦的事。在jdk1.3以后,jdk跟我们提供了一个API java.lang.reflect.InvocationHandler的类, 这个类可以让我们在JVM调用某个类的方法时动态的为些方法做些什么事。下面我们就来实现动态代理的实现。

  动态代理实现主要是实现InvocationHandler,并且将目标对象注入到代理对象中,利用反射机制来执行目标对象的方法。

  接口实现与静态代理相同,代理类代码:

1 public class DynaProxyHello implements InvocationHandler{
 2     
 3     private Object target;//目标对象
 4     /**
 5      * 通过反射来实例化目标对象
 6      * @param object
 7      * @return
 8      */
 9     public Object bind(Object object){
10         this.target = object;
11         return Proxy.newProxyInstance(this.target.getClass().getClassLoader(), this.target.getClass().getInterfaces(), this);
12     }
13     
14     @Override
15     public Object invoke(Object proxy, Method method, Object[] args)
16             throws Throwable {
17         Object result = null;
18         Logger.start();//添加额外的方法
19         //通过反射机制来运行目标对象的方法
20         result = method.invoke(this.target, args);
21         Logger.end();
22         return result;
23     }
24     
25 }

 

 

 

 

 

### 回答1: Spring AOPSpring框架的一部分,主要用于在应用程序的不同部分之间进行面向切面编程(AOP)。AOP是一种编程范式,可以将应用程序的关注点分离,使应用程序更具可重用性和可维护性。 Spring AOP使用代理模式在运行时织入额外的功能,可以在方法调用前后、异常处理等方面添加额外的逻辑。使用Spring AOP可以将应用程序的关注点(如事务处理、日志记录等)分离出来,从而提高应用程序的可维护性和可重用性。 Spring AOP支持使用AspectJ注解和XML配置两种方式来定义切面。Spring AOP还提供了一个强大的表达式语言(Spring Expression Language,简称SpEL),用于在运行时动态地匹配连接点。 总的来说,Spring AOP是一个非常强大的AOP框架,可以帮助我们更好的管理应用程序的关注点,提高应用程序的可维护性和可重用性。 ### 回答2: Spring AOP(面向切面编程)是Spring框架的一个核心模块,用于实现跨应用程序模块的关注点分离。它通过在系统中将横切关注点从业务逻辑中分离出来,实现了代码的重用性、可维护性和扩展性。 Spring AOP基于代理模式,通过动态代理技术在运行时将通用的横切关注点织入目标对象中,从而实现了对目标对象方法的拦截和增强。Spring AOP主要通过使用切面、切点、通知和连接点等概念来实现。 - 切面(Aspect):切面是一个模块,它包含了一组与横切关注点相关的通知和切点。通常,一个应用程序由多个切面组成,每个切面负责某个特定的关注点。比如,日志记录、性能监控等都可以作为切面。 - 切点(Pointcut):切点定义了在哪些目标对象方法上应用通知。它使用表达式来匹配目标对象中的方法。比如,通过使用通配符等方式,我们可以定义一个切点来匹配所有的Service层方法。 - 通知(Advice):通知定义了在切点上执行的行为。Spring AOP提供了五种类型的通知:前置通知(Before)、后置通知(AfterReturning)、异常通知(AfterThrowing)、最终通知(After)和环绕通知(Around)。通过在切面中配置不同类型的通知,我们可以在目标对象方法的不同执行阶段执行相应的操作。 - 连接点(Joinpoint):连接点是在目标对象中匹配到切点的特定位置,比如方法的调用、方法的返回等。在连接点上,可以执行相应的通知。 Spring AOP的主要优势是它能够将关注点分离,使开发人员更关注业务逻辑而不是横切关注点的实现。通过使用Spring AOP,我们可以实现日志记录、事务管理、性能监控等与业务逻辑无关的功能,增加了代码的可维护性和可重用性。此外,Spring AOP还提供了灵活的配置方式,可以使用XML配置或基于注解的方式来定义切面和通知,使代码更加简洁和易于理解。 总之,Spring AOPSpring框架中实现面向切面编程的核心模块,通过在运行时动态生成代理对象,实现了对目标对象方法的拦截和增强。它通过将通用的横切关注点从业务逻辑中抽离出来,提高了代码的可维护性和可重用性。 ### 回答3: Spring AOP(Aspect-Oriented Programming)是Spring框架中的一个重要模块,用于实现面向切面编程的功能。它通过在不修改原有代码的情况下,将横切关注点(如日志记录、安全检查、事务管理等)从业务逻辑中分离出来,从而提高系统的可维护性和可重用性。 Spring AOP的核心原理是基于动态代理技术,通过在运行时动态生成代理对象来实现AOP功能。它主要由以下几个核心概念组成: 1. 切面(Aspect):是对一个特定横切关注点的抽象描述,它包含了一组通知(Advice)和一个切点(Pointcut)。通知定义了在切点处执行的具体操作,切点定义了在应用程序中哪些位置应该触发通知。 2. 连接点(Join point):是在程序运行过程中,符合切点定义的特定位置,如方法调用、异常抛出等。Spring AOP仅支持方法级别的切点。 3. 通知(Advice):是在连接点处执行的具体操作,包括前置通知(Before)、后置通知(After)、返回通知(AfterReturning)、异常通知(AfterThrowing)和环绕通知(Around)等。它们分别在连接点的不同位置执行,实现不同的横切关注点功能。 4. 切点表达式(Pointcut Expression):是一个表达式,用于匹配特定的切点位置。Spring AOP支持使用AspectJ风格的切点表达式语法。 Spring AOP的应用步骤如下: 1. 定义切面类,包含通知和切点的定义。 2. 配置切面,通过XML配置或基于注解的方式将切面配置到Spring容器中。 3. 定义目标类,即需要被代理的类。 4. 配置代理,通过XML配置或基于注解的方式将目标类与切面关联起来。 5. 启动Spring容器,自动完成代理的实例化和连接点的绑定。 6. 运行程序,当连接点满足切点定义时,代理会根据定义的通知类型执行相应的操作。 总之,Spring AOP是一种强大的横切关注点分离工具,它通过代理机制实现对目标类的增强,使得系统的关注点分离更加灵活和简单。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值