首先我们来看一下登录的原理
如上图所示这是一个基本的登录原理图,但是如果我们想要在这个登录之上添加一些新的功能,比如权限校验
那么我们能想到的就有两种方法:
①:通过对源代码的修改实现
②:不通过修改源代码方式添加新的功能 (AOP)
AOP相关的概念 1. AOP的概述什么是AOP的技术?
在软件业,AOP为Aspect Oriented Programming的缩写,意为:面向切面编程。
AOP是一种编程范式,隶属于软工范畴,指导开发者如何组织程序结构。
AOP最早由AOP联盟的组织提出的,制定了一套规范.Spring将AOP思想引入到框架中,必须遵守AOP联盟的规范。
通过预编译方式或者运行期动态代理实现程序功能的统一维护的一种技术。
AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。
利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
AOP采取横向抽取机制,取代了传统纵向继承体系重复性代码(事务管理、安全检查、缓存)
为什么要学习AOP,可以在不修改源代码的前提下,对程序进行增强!!
2. AOP的优势运行期间,不修改源代码的情况下对已有的方法进行增强。
优势:
- 减少重复的代码
- 提供开发的效率
- 维护方便
JDK的动态代理技术
- 为接口创建代理类的字节码文件
- 使用ClassLoader将字节码文件加载到JVM
- 创建代理类实例对象,执行对象的目标方法
cglib代理技术
为类生成代理对象,被代理类有没有接口都无所谓,底层是生成子类,继承被代理类。
AOP底层实现原理:(核心就是增强)
JDK动态代理(必须有接口),生成代理对象实现相同的接口JDK动态代理底层采用接口的方式,实现的增强。
CGLIB代理技术对类生成代理对象被代理的类是否实现接口。无所谓CGLIB底层是采用类继承的方式,实现的增强。
Joinpoint(连接点) 类里面有哪些方法可以增强这些方法称为连接点
Pointcut(切入点) -- 所谓切入点是指我们要对哪些Joinpoint进行拦截的定义
Advice(通知/增强)-- 所谓通知是指拦截到Joinpoint之后所要做的事情就是通知.通知分为前置通知,后置通知,异常通知,最终通知,环绕通知(切面要完成的功能)
Aspect(切面)-- 是 切入点+通知 的结合,以后咱们自己来编写和配置的
1.连接点
类里面有哪些方法可以增强这些方法称为连接点,右边的几个方法当中谁可以被增强谁就是连接点。
2.切入点
实际被增强的方法就是切入点,比如这四个方法都可以增强,但是我们只增强add()方法,那么add()方法就是切入点。
3.通知(增强)
(1)实质增强的逻辑部分称为通知,就比如我们讲的登录逻辑当中的权限判断就是通知(增强)
(2)通知有多种类型:
- 前置通知:在一个方法之前执行
- 后置通知:在一个方法执行完之后执行
- 环绕通知:在一个方法执行之前和之后都执行
- 异常通知:当一个方法发生异常的时候执行
- 最终通知:类似于finally最后永选执行
4.切面:是一个动作,将通知应用到切入点的过程
2. 基本准备工作
AspectJ是一个面向切面的框架,它扩展了Java语言。AspectJ定义了AOP语法,AspectJ实际上是对AOP编程思想的一个实践.
2. AOP配置文件方式的入门创建maven项目,坐标依赖
org.springframework
spring-context
5.0.2.RELEASE
commons-logging
commons-logging
1.2
org.springframework
spring-test
5.0.2.RELEASE
log4j
log4j
1.2.12
junit
junit
4.12
test
aopalliance
aopalliance
1.0
org.springframework
spring-aspects
5.0.2.RELEASE
org.aspectj
aspectjweaver
1.8.3
创建被增强的类
// 被增强的类
public class User {
//连接点/切入点
public void add(){
System.out.println("add......");
}
public void update(){
System.out.println("update......");
}
}
将目标类配置到Spring中
定义切面类
public class UserProxy {
//增强/通知 ---》前置通知
public void before(){
System.out.println("before.............");
}
}
在配置文件中定义切面类
在配置文件中完成aop的配置
完成测试
public class DemoTest {
@Test
public void aopTest1(){
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
User user = (User) applicationContext.getBean("user");
user.add();
}
}
3. 切入点的表达式
再配置切入点的时候,需要定义表达式,具体展开如下:
切入点表达式的格式如下:
execution([修饰符] [返回值类型] [类全路径] [方法名 ( [参数] )])
修饰符可以省略不写,不是必须要出现的。
返回值类型是不能省略不写的,根据你的方法来编写返回值,可以使用 * 代替。
包名,类名,方法名,参数的规则如下:
例如:com.qcby.demo3.BookDaoImpl.save()
首先包名,类名,方法名是不能省略不写的,但是可以使用 * 代替
中间的包名可以使用 * 号代替
类名也可以使用 * 号代替,也有类似的写法:*DaoImpl
方法也可以使用 * 号代替
参数如果是一个参数可以使用 * 号代替,如果想代表任意参数使用 ..
比较通用的表达式:execution(* com.qcby.*.ServiceImpl.save(..))
举例2:com.qcby.demo3.BookDaoImpl当中所有的方法进行增强
execution(* com.qcby.*.ServiceImpl.*(..))
举例3:com.qcby.demo3包当中所有的方法进行增强
execution(* com.qcby.*.*.*(..))
4. AOP的通知类型
1. 前置通知 目标方法执行前,进行增强。
如上配置案例就是前置通知
2. 环绕通知 目标方法执行前后,都可以进行增强。目标对象的方法需要手动执行。
// 环绕通知
public void around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
System.out.println("before.............");
// 执行被增强的方法
proceedingJoinPoint.proceed();
System.out.println("after.............");
}
xml配置
3. 最终通知 目标方法执行成功或者失败,进行增强。
// 最终通知
public void after() {
System.out.println("after.............");
}
xml配置
4. 后置通知 目标方法执行成功后,进行增强。
//后置通知
public void afterReturning() {
System.out.println("afterReturning.............");
}
xml配置
5. 异常通知 目标方法执行失败后,进行增强。(发生异常的时候才会执行,否则不执行)
//异常通知
public void afterThrowing() {
System.out.println("afterThrowing.............");
}
需要改动一下切点
//连接点/切入点
public void add(){
int a = 10 / 0;
System.out.println("add......");
}
xml配置
Spring的AOP技术-注解方式
1. AOP注解方式入门程序
创建maven工程,导入坐标。编写接口,完成IOC的 *** 作。步骤略。
编写切面类
给切面类添加注解 @Aspect,编写增强的方法,使用通知类型注解声明
1.配置xml扫描注解
2.配置注解
@Component
public class User {
//连接点/切入点
public void add(){
System.out.println("add......");
}
}
给切面类添加注解 @Aspect,编写增强的方法,使用通知类型注解声明
@Component
@Aspect //生成代理对象
public class UserProxy {
}
3.配置文件中开启自动代理
4.通知类型注解
@Before -- 前置通知
@AfterReturing -- 后置通知
@Around -- 环绕通知(目标对象方法默认不执行的,需要手动执行)
@After -- 最终通知
@AfterThrowing -- 异常抛出通知
@Component
@Aspect //生成代理对象
public class UserProxy {
//增强/通知 ---》前置通知
@Before(value = "execution(* com.*.User.add(..))")
public void before(){
System.out.println("before.............");
}
// 环绕通知
@Around(value = "execution(* com.*.User.add(..))")
public void around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
System.out.println("before.............");
// 执行被增强的方法
proceedingJoinPoint.proceed();
System.out.println("after.............");
}
// 最终通知
@After(value = "execution(* com.*.User.add(..))")
public void after() {
System.out.println("after.............");
}
//异常通知
@AfterThrowing(value = "execution(* com.*.User.add(..))")
public void afterThrowing() {
System.out.println("afterThrowing.............");
}
//后置通知
@AfterReturning(value = "execution(* com.*.User.add(..))")
public void afterReturning() {
System.out.println("afterReturning.............");
}
}
5.测试类
@Test
public void aopTest1(){
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
user.add();
}
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)