一、AOP
AOP:AspectOriented Programming,意为:面向切面编程。是OOP(面向对象编程)的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
AOP采取横向抽取机制,取代了传统纵向继承体系重复性代码。
经典应用:事务管理、性能监视、安全检查、缓存 、日志等。
Spring中的 AOP使用纯Java实现,不需要专门的编译过程和类加载器,在运行期通过代理方式向目标类织入增强代码。
aop底层将采用代理机制进行实现:
接口 + 实现类 :spring采用jdk的动态代理Proxy。
实现类:spring采用cglib字节码增强。
1、AOP术语(掌握)
1. target:目标类,需要被代理的类。例如:UserService。
2. Joinpoint:(连接点):所谓连接点是指那些可能被拦截到的方法。例如:所有的方法
3. PointCut:切入点:已经被增强的连接点。例如:addUser()
4. advice:通知/增强,增强代码。例如:after、before。
5. Weaving:(织入):是指把增强advice应用到目标对象target来创建新的代理对象proxy的过程.
6. proxy:代理类
7. Aspect:(切面): 是切入点pointcut和通知advice的结合
一个线是一个特殊的面。
一个切入点和一个通知,组成成一个特殊的面。
2、手动方式实现代理
JDK动态代理
(1). 创建目标类
使用jdk动态代理,要求目标类必须 接口+实现类 。
(2). 创建切面类
(3). 创建工厂类
(4). 测试
CGLIB字节码增强
如果目标类没有实现接口,只有类,CGLIB进行字节码增强。
需要cglib.jar和asm.jar包,spring的core.jar包中已经集成了需要的jar包。
(1). 创建目标类
(2). 创建切面类
(3). 创建工厂类
(4). 测试
3. AOP联盟通知类型
AOP联盟为通知Advice定义了org.aopalliance.aop.AdviceSpring按照通知在目标类方法的连接点位置,可以分为5类
前置通知 org.springframework.aop.MethodBeforeAdvice
在目标方法执行前实施增强
后置通知 org.springframework.aop.AfterReturningAdvice
在目标方法执行后实施增强
环绕通知 org.aopalliance.intercept.MethodInterceptor,必须手动执行目标方法
在目标方法执行前后实施增强
异常抛出通知 org.springframework.aop.ThrowsAdvice
在方法抛出异常后实施增强
引介通知 org.springframework.aop.IntroductionInterceptor
在目标类中添加一些新的方法和属性
4. spring全自动实现代理(AspectJ)
AspectJ是一个基于Java语言的AOP框架。Spring2.0以后新增了对AspectJ切入点表达式支持。@AspectJ 是AspectJ1.5新增功能,通过JDK5注解技术,允许直接在Bean类中定义切面。
(1)切入点表达式(掌握)
指定目标类中需要增强的方法,可以用通配符execution()用于描述方法(掌握)
语法:execution(修饰符 返回类型 包.类.方法名(参数) throws 异常)
修饰符:一般省略(public、*)
返回值:不能省略(void、… …、*)
包:[可以省略]
com.a 固定包
com.a.*.service a包下面子包任意
com.a.b.. b包下面的所有子包(含自己)
com.a.b.*.c.. b包下面任意子包,固定目录c,c目录任意包
类:[可以省略]
UserServiceImpl 指定类
*Impl 以Impl结尾
User* 以User开头
* 任意
方法名:不能省略
addUser 固定方法
add* 以add开头
*Do 以Do结尾
* 任意
(参数):
() 无参
(int ,int) 两个
(..) 参数任意
throws :可省略,一般不写。
综合1
execution(* com.a.b.*.c..*.*(..))
AspectJ通知类型
aop联盟定义通知类型,具有特定接口,必须实现特定通知类型接口,重写方法。aspectj通知类型,只定义类型名称。已及方法格式。
before:前置通知(应用:各种校验)
在方法执行前执行,如果通知抛出异常,阻止方法运行
afterReturning:后置通知(应用:常规数据处理)
方法正常返回后执行,如果方法中抛出异常,通知无法执行
必须在方法执行后才执行,所以可以获得方法的返回值。
around:环绕通知(应用:十分强大,可以做任何事情)
方法执行前后分别执行,可以阻止方法的执行
必须手动执行目标方法
afterThrowing:抛出异常通知(应用:包装异常信息)
方法抛出异常后执行,如果方法没有抛出异常,无法执行
after:最终通知(应用:清理现场)
方法执行完毕后执行,无论方法中是否出现异常
(2)导入jar包
spring的基础包:4+1aop相关的包:aopalliance-1.0.jar 和 spring-aop-4.1.0.RELEASE.jar
aspectj相关的包:aspectjweaver-1.7.4.jar 和 spring-aspects-4.1.0.RELEASE.jar
(3)基于xml配置(掌握)
a. 目标类:接口+实现类
b. 切面类
c. spring配置
(4)基于注解配置(掌握)
bean替换
a. 把业务层UserService的bean定义替换成:
b. 把切面类的bean定义
替换成:
切面类内容替换
a. 把切面定义
替换成:
b. 把前置通知配置
替换成:
c. 把全局切入点表达式
替换成:
d. 同理替换后置、环绕、异常、最终通知
spring配置
配置bean的自动扫描、和切面注解的有效
总结
@Aspect 声明类是切面类,从而获得通知@Before 声明前置通知
@PointCut 声明全局切入点表达式,修改方法,用方法名引用切入点
@AfterReturning 声明后置通知
@Around 声明环绕通知
@AfterThrowing 声明异常通知
@After 声明最终通知
5. 事务管理
(1) 基础知识
事务:一组不可分割的多条sql语句的集合事务特性:ACID
事务并发问题:
1. 幻读:T1查询,T2插入或删除,T1再查,发现数据少或多。
2. 不可重复读:T1查询一条,T2修改此数据,T1再查,发现不同。
3. 脏读:T1修改数据,T2读取数据,T1回滚,T2读取的数据是错的。
隔离级别:
ReadUncommitted:读未提交
ReadCommitted:读已提交
ReadRepeatable:可重复读
Serializable:串行读
详细请看:https://blog.csdn.net/qq_27339781/article/details/80249907
(2)事务管理简介
PlatformTransactionManager 事务管理器
事务管理器:如果要进行事务管理,必须配置事务管理器。常用的事务管理器有:
DataSourceTransactionManager:是jdbc的事务管理器类
HibernateTransactionManager:是hibernamte的事务管理器类(spring-orm.jar)
API:
TransactionStatus getTransaction(TransactionDefinition definition):根据详情获得状态
commit(TransactionStatus status):提交事务
rollback(TransactionStatus status):回滚事务
TransactionStatus 事务状态
包含了事务是否是新建的、有无保存点等状态。TransactionDefinition 事务详情
包含事务的隔离级别、传播行为等信息。配置事务时,必须配置事务详情事务的传播行为
需要掌握:PROPAGATION_REQUIRED、PROPAGATION_ REQUIRED_NEW、PROPAGATION_NESTED
(3)AOP基于xml配置事务
a. 配置事务管理器
b. 配置事务详情
c. 配置aop应用事务
(4)AOP基于注解配置事务
a. 配置事务管理器
b. 把事务管理器给spring自动管理
c. 在service上用@Transaction标注类