目录
(一)Spring Aop基本介绍
1.含义
AOP(Aspect-OrientedProgramming,面向切面编程),可以说是OOP(Object-Oriented Programing,面向对象编程)的补充和完善。OOP引入封装、继承和多态性等概念来建立一种对象层次结构,用以模拟公共行为的一个集合。当我们需要为分散的对象引入公共行为的时候,OOP则显得无能为力。也就是说,OOP允许你定义从上到下的关系,但并不适合定义从左到右的关系。例如日志功能。日志代码往往水平地散布在所有对象层次中,而与它所散布到的对象的核心功能毫无关系。对于其他类型的代码,如安全性、异常处理和透明的持续性也是如此。这种散布在各处的无关的代码被称为横切(cross-cutting)代码,在OOP设计中,它导致了大量代码的重复,而不利于各个模块的重用。而AOP技术则恰恰相反,它利用一种称为“横切”的技术,剖解开封装的对象内部,并将那些影响了多个类的公共行为封装到一个可重用模块,并将其名为“Aspect”,即方面。所谓“方面”,简单地说,就是将那些与业务无关,却为业务模块所共同调用的逻辑或责任封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于未来的可操作性和可维护性。AOP代表的是一个横向的关系,如果说“对象”是一个空心的圆柱体,其中封装的是对象的属性和行为;那么面向方面编程的方法,就仿佛一把利刃,将这些空心圆柱体剖开,以获得其内部的消息。而剖开的切面,也就是所谓的“方面”了。然后它又以巧夺天功的妙手将这些剖开的切面复原,不留痕迹。
2.基本作用
1.软件开发原则:高内聚,低耦合
2.Spring的AOP作用在于解耦。AOP让一组类共享相同的行为(比如事务管理、日志管理、安全管理)
3.OOP(Object-Oriented Programming)只能通过继承类或实现接口来增加代码的耦合度,而且类继承是单根继承(不允许一子多父),阻碍了将更多的行为添加到一组类上,此时AOP可以弥补OOP的不足。
3.Aop和Oop
AOP(Aspect-Oriented Programming)—— 横向的关系
OOP(Object-Oriented Programming)—— 纵向的关系
4.Aop使用方式
- Spring采用配置方式使用AOP
- Spring采用注解方式使用AOP
5.Aop基本概念
- Aspect(切面):通常是一个类,里面可以定义切入点和通知
- JointPoint(连接点):程序执行过程中明确的点,一般是方法的调用
- Advice(通知):AOP在特定的切入点上执行的增强处理,有before、after、after-returning、after-throwing、around
- Pointcut(切入点):就是带有通知的连接点,在程序中主要体现为书写切入点表达式
- AOP代理:AOP框架创建的对象,代理就是目标对象的加强。Spring中的AOP代理可以使JDK动态代理,也可以是CGLIB代理,前者基于接口,后者基于子类
(二)配置方式使用AOP
1.创建aop.xml包
2.创建杀龙任务类
package net.hw.spring.lesson5.aop_xml;
import org.springframework.stereotype.Component;
/**
* @author zoulei
* @create_time 2021-03-24-8:21
*/
@Component
public class SlayDragonQuest {
public void embark(){
System.out.println("执行杀龙任务");
}
}
3.创建骑士类
package net.hw.spring.lesson5.aop_xml;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
/**
* @author zoulei
* @create_time 2021-03-24-8:22
*/
@Component("Mike")
public class BraveKnight {
@Autowired
private SlayDragonQuest slayDragonQuest;
public void embarkOnQuest(){
slayDragonQuest.embark();
}
}
4.创建诗人类
package net.hw.spring.lesson5.aop_xml;
import org.springframework.stereotype.Component;
/**
* 游呤诗人
*
* @author zoulei
* @create_time 2021-03-31-11:21
*/
@Component
public class Minstrel {
public void singBeforeQuest(){
System.out.println("啦啦啦 骑士出发了!");
}
public void singAfterQuest(){
System.out.println("恭喜你,完成任务!");
}
}
5.创建Spring配置文件
在resources下创建aop_xml,并且创建spring-config.xml配置文件
<?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:context="http://www.springframework.org/schema/context"
xmlns:aop="http://www.springframework.org/schema/aop"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop https://www.springframework.org/schema/aop/spring-aop.xsd">
<!-- 组件扫码-->
<context:component-scan base-package="net.hw.spring.lesson5.aop_xml"/>
<!--AOP配置-->
<aop:config>
<!--定义切面-->
<aop:aspect ref="minstrel">
<!--定义切点 -->
<aop:pointcut id="emabrk" expression="execution(* net.hw.spring.lesson5..*.embarkOnQuest(..))"/>
<!--声明前置通知-->
<aop:before method="singBeforeQuest" pointcut-ref="emabrk"/>
<!--声明后置通知-->
<aop:after method="singBeforeQuest" pointcut-ref="emabrk"/>
</aop:aspect>
</aop:config>
</beans>
1.切点
在使用Spring框架配置AOP时,不管是通过XML配置文件还是注解方式,都需要定义pointcut(切点)。
2.切点表达式
execution()是最常用的切点函数,整个表达式可以分为五个部分。
- execution():表达式主体。
- 第一个*号:表示返回类型,*号表示所有的类型。
- 包名:表示需要拦截的包名,后面的两个句点表示当前包和当前包的所有子包,net.zl.spring.lesson05包、子孙包下所有类的方法。
- 第二个*号:表示类名,*号表示所有的类。
- *(…):最后这个星号表示方法名,*号表示所有的方法,后面括弧里面表示方法的参数,两个句点表示任何参数。
6.在pom文件添加AOP相关依赖
<!--Spring AOP-->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aop</artifactId>
<version>${spring.version}</version>
</dependency>
<!--AspectJ支持-->
<dependency>
<groupId>aspectj</groupId>
<artifactId>aspectjrt</artifactId>
<version>1.5.4</version>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.9.6</version>
<scope>runtime</scope>
</dependency>
创建测试类
package net.hw.spring.lesson05;
import net.hw.spring.lesson5.aop_xml.BraveKnight;
import net.hw.spring.lesson5.aop_xml.DamseLRescuingKnight;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
/**
* @author zoulei
* @create_time 2021-03-31-10:19
*/
public class TestKnight {
private AnnotationConfigApplicationContext context;
@Before
public void init(){
//基于注解配置类创建应用容器
context = new AnnotationConfigApplicationContext("aop_xml/spring-config.xml");
}
@Test
public void testBraveKnight(){
//根据名称从应用容器里获取勇敢类对象
BraveKnight knight = (BraveKnight) context.getBean("Mike");
//骑士执行杀龙任务
knight.embarkOnQuest();
}
@Test
public void testDamseLRecuingKnight(){
DamseLRescuingKnight knight = (DamseLRescuingKnight)context.getBean("damseLRescuingKnight");
knight.embarkOnQuest();
}
@After
public void destroy(){
//关闭应用容器
context.close();
}
}
完成测试
三.用注解方式使用AOP
1.创建类
package net.hw.spring.lesson5.aop_annotation;
import org.springframework.stereotype.Component;
/**
* @author zoulei
* @create_time 2021-03-24-8:21
*/
@Component
public class SlayDragonQuest {
public void embark(){
System.out.println("执行杀龙任务");
}
}
package net.hw.spring.lesson5.aop_annotation;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
/**
* @author zoulei
* @create_time 2021-03-24-8:22
*/
@Component("Mike")
public class BraveKnight {
@Autowired
private SlayDragonQuest slayDragonQuest;
@Action(name = "动作拦截")
public void embarkOnQuest(){
slayDragonQuest.embark();
}
}
2.创建切面
package net.hw.spring.lesson5.aop_annotation;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.After;
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;
/**
* 游呤诗人
*
* @author zoulei
* @create_time 2021-03-31-11:21
*/
@Component
public class Minstrel {
// 注解声明切点
@Pointcut("@annotation(net.hw.spring.lesson5.aop_annotation.Action)")
public void embark() {
}
// 注解声明前置通知
@Before("embark()")
public void singBeforeQuest(JoinPoint joinPoint) {
// 获取方法签名
MethodSignature signature = (MethodSignature) joinPoint.getSignature();
// 获取被拦截的方法
Method method = signature.getMethod();
// 获取注解式拦截
Action action = method.getAnnotation(Action.class);
// 提示用户被拦截了
System.out.println("[" + action.name() + "]拦截了[" + method.getName() + "]:拦截前!");
System.out.println("啦啦啦,骑士出发了!");
}
// 注解声明后置通知
@After("embark()")
public void singAfterQuest(JoinPoint joinPoint) {
// 获取方法签名
MethodSignature signature = (MethodSignature) joinPoint.getSignature();
// 获取被拦截的方法
Method method = signature.getMethod();
// 获取注解式拦截
Action action = method.getAnnotation(Action.class);
// 提示用户被拦截了
System.out.println("[" + action.name() + "]拦截了[" + method.getName() + "]:拦截后!");
System.out.println("真棒啊!骑士完成了任务!");
}
}
3.创建配置类
package net.hw.spring.lesson5.aop_annotation;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
/**
* @author zoulei
* @create_time 2021-04-07-8:45
*/
@Configuration//标明spring标志类
@ComponentScan("net.hw.spring.lesson5.aop_annotation")//组件扫码
@EnableAspectJAutoProxy//开启Spring对AspectJ的支持
public class AopConfig {
}
4.创建测试类
package net.hw.spring.lesson05.aop_annotation;
import net.hw.spring.lesson5.aop_annotation.AopConfig;
import net.hw.spring.lesson5.aop_annotation.BraveKnight;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
/**
* @author zoulei
* @create_time 2021-04-07-8:49
*/
public class TestKnight {
private AnnotationConfigApplicationContext context; //基于注解配置类的应用容器
@Before
public void init(){
//基于注解配置类创建容器
context = new AnnotationConfigApplicationContext(AopConfig.class);
}
@Test
public void testBraveKnight(){
//根据名称从应用容器获取勇敢骑士类对象
BraveKnight knight = (BraveKnight) context.getBean("Mike");
//勇敢骑士执行任务
knight.embarkOnQuest();
}
@After
public void destroy(){
//关闭容器
context.close();
}
}
测试结果