分析

AOP: 面向切面编程,相对于OOP面向对象编程。

OOP: Object Oriented Programming,面向对象的程序设计。

Spring的AOP的存在目的是为了解耦。AOP可以让一组类共享相同的行为。

在OOP中只能通过继承类和实现接口,来使代码的耦合度增强,且类继承只能为单继承,阻碍更多行为添加到一组类上,AOP弥补了OOP的不足。

Spring支持AspectJ的注解式切面编程

1、使用@Aspect声明类是一个切面
2、使用@After,@Before,@Around 定义建言(advice),可直接将拦截规则(切点)作为参数。
3、其中@After,@Before,@Around参数的拦截规则为切点(PointCut),
为了使切点复用,可使用@PointCut专门定义拦截规则,然后在@After,@Before,@Around的参数中调用
4、其中符合条件的每一个被拦截处为连接点(JoinPoint)

本示例演示基于注解拦截和基于方法拦截两种方式,演示一种模拟记录操作的日志系统的实现。

其中注解式拦截能够很好地控制要拦截的粒度和获得更丰富的信息,Spring本身在事务处理(@Transcational)和数据缓存(@Cacheable等)上面都使用此种形式的拦截。

pom.xml的配置不再累赘写上。

不知道写的朋友请见此篇博客:


示例

添加Spring aop支持及AspectJ依赖

<!-- spring aop 支持-->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aop</artifactId>
<version>4.2.3.RELEASE</version>
</dependency>

<!-- aspectj支持 -->
<!-- https://mvnrepository.com/artifact/org.aspectj/aspectjrt -->
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjrt</artifactId>
<version>1.8.9</version>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.8.9</version>
</dependency>

编写拦截规则的注解

package cn.hncu.p1_3_3_aop;

import java.lang.annotation.*;

/**
* Created with IntelliJ IDEA.
* User: 陈浩翔.
* Date: 2016/11/9.
* Time: 上午 11:11.
* Explain:注解类
*/
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Action {
String name();
}

编写使用注解的被拦截类

package cn.hncu.p1_3_3_aop;

import org.springframework.stereotype.Service;

/**
* Created with IntelliJ IDEA.
* User: 陈浩翔.
* Date: 2016/11/9.
* Time: 上午 11:16.
* Explain:使用注解的被拦截类
*/
@Service
public class DemoAnnotationService {
@Action(name = "@Action---DemoAnnotationService.add操作")
public void add(){
System.out.println("DemoAnnotationService.add...");
}
}

编写使用方法规则的被拦截类

package cn.hncu.p1_3_3_aop;

import org.springframework.stereotype.Service;

/**
* Created with IntelliJ IDEA.
* User: 陈浩翔.
* Date: 2016/11/9.
* Time: 上午 11:23.
* Explain:使用方法规则的被拦截类
*/
@Service
public class DemoMethodService {
@Action(name="@Action---DemoMethodService.add操作")
public void add(){
System.out.println("DemoMethodService.add...");
}
}

编写切面

package cn.hncu.p1_3_3_aop;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.stereotype.Component;

import java.lang.reflect.Method;

/**
* Created with IntelliJ IDEA.
* User: 陈浩翔.
* Date: 2016/11/9.
* Time: 上午 11:24.
* Explain:切面
*/
@Aspect
//通过@Aspect注解声明这是一个切面
@Component
//通过@Component让此切面成为Spring容器管理的Bean
public class LogAspect {

//注意这两个@Pointcut写法的区别!!!
// 一个是拦截注解(写了@Action注解的方法都会被拦截),一个是拦截类方法
@Pointcut("@annotation(cn.hncu.p1_3_3_aop.Action)")//通过@PointCut注解声明切点
//@Pointcut("execution(* cn.hncu.p1_3_3_aop.DemoAnnotationService..*(..))")
//配置切入点,该方法无方法体,主要为方便同类中其他方法使用此处配置的切入点
public void annotatiomPointCut(){ }

//拦截注解
@After("annotatiomPointCut()")//通过@After注解声明一个建言,并使用@PointCut定义的切点
public void after(JoinPoint joinPoint){
MethodSignature signature =(MethodSignature)joinPoint.getSignature();
Method method = signature.getMethod();
Action action = method.getAnnotation(Action.class);
System.out.println(action.name());//通过反射可获得注解上的属性,可以用来做日志记录等相关操作
}

@Before("execution(* cn.hncu.p1_3_3_aop.DemoMethodService.*(..))")
//通过@Before注解声明一个建言,此建言直接使用拦截规则作为参数
public void before(JoinPoint joinPoint){
MethodSignature signature = (MethodSignature) joinPoint.getSignature();
Method method = signature.getMethod();
Action action = method.getAnnotation(Action.class);
System.out.println("方法规则式拦截:" + method.getName()+" "+action.name());
}


}

编写配置类

package cn.hncu.p1_3_3_aop;

import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.EnableAspectJAutoProxy;

/**
* Created with IntelliJ IDEA.
* User: 陈浩翔.
* Date: 2016/11/9.
* Time: 上午 11:42.
* Explain:配置类
*/
@Configuration
@ComponentScan("cn.hncu.p1_3_3_aop")
@EnableAspectJAutoProxy //使用@EnableAspectJAutoProxy注解开启Spring对AspectJ代理的支持
public class AopConfig {
}

运行类

package cn.hncu.p1_3_3_aop;

import org.springframework.context.annotation.AnnotationConfigApplicationContext;

/**
* Created with IntelliJ IDEA.
* User: 陈浩翔.
* Date: 2016/11/9.
* Time: 上午 11:41.
* Explain:运行类
*/
public class Main {
public static void main(String[] args) {
AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(AopConfig.class);
DemoAnnotationService demoAnnotationService = context.getBean(DemoAnnotationService.class);
demoAnnotationService.add();

DemoMethodService demoMethodService = context.getBean(DemoMethodService.class);
demoMethodService.add();

context.close();
}
}

运行结果

【Spring】Spring基础配置-AOP_spring

项目链接—具体包:

​https://github.com/chenhaoxiang/Java/tree/master/springBoot/src/main/java/cn/hncu/p1_3_3_aop​


作者:谙忆, 所有权利保留。