分析
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();
}
}
运行结果
项目链接—具体包:
https://github.com/chenhaoxiang/Java/tree/master/springBoot/src/main/java/cn/hncu/p1_3_3_aop
作者:谙忆, 所有权利保留。