[Spring+SpringMVC+Mybatis]框架学习笔记(四):Spring实现AOP
上一章:[Spring+SpringMVC+Mybatis]框架学习笔记(三):Spring实现JDBC
下一章:[Spring+SpringMVC+Mybatis]框架学习笔记(五):SpringAOP_顾问
第4章 Spring实现AOP
4.1 AOP的概念
AOP: Aspect Oriented Programming 面向切面编程
AOP的实现可以把业务逻辑和系统级的服务进行隔离,使业务逻辑跟各个系统级服务的耦合度降低,提高系统的重用性和开发效率。
业务逻辑:某个类的某个方法本身要实现的功能。
系统级的服务:系统的日志、事务、权限验证
举例:六七十年代,村支书要通知一个事情,原来要一家一家逐个告诉他们;有了广播(AOP)以后,只需要用广播,大家就知道了。
4.2 AOP的实现
AOP底层实现的原理:动态代理
动态代理有两种方式:
- jdk动态代理(只能代理实现了接口的类)
- cglib动态代理(都可以) ----> 字节码生成器
4.3 AOP的一些专业术语
- 切面(Aspect):交叉在各个业务逻辑中的系统服务,例如:安全验证、事务处理、日志记录等
- 织入(weaving):将切面代码插入到目标对象的某个方法的过程,相当于jdk动态代理中的InvacationHandler接口方法的内容。
- 连接点(JointPoint):理论上可以被切面织入的所有方法,例如addStudent(),delStudent()...通常所有方法都可以被称为连接点。
- 切入点(PointCut): 实际上被切面织入的方法。
- 目标对象(target):就是切入点所属的类,如:StudentService类
- 通知(Advice):就是切面的实现,切面织入的目标对象,方法,时间点(方式执行前、后、前后、异常)以及内容。
- 顾问(Advisor):就是通知的封装和延伸。可以将通知以更复杂的方式织入到某些方法中。
把前5个术语串联起来:
将切面织入到目标对象的连接点方法中,使连接点成为切入点。
4.4 Spring实现AOP的步骤
1)搭建SpringAOP开发环境需要引入的jar包;
aopalliance-1.0.jar
spring-aop-4.2.1.RELEASE.jar
2)实现通知
- 前置通知 implements org.springframework.aop.MethodBeforeAdvice
- 后置通知 implements org.springframework.aop.AfterReturningAdvice
- 环绕通知 implements org.aopalliance.intercept.MethodInterceptor
- 异常通知 implements org.springframework.aop.ThrowsAdvice
3)注册目标类
4)注册通知
5)注册代理生产器,并注入目标类和通知
ps:通知的注意点:
- 代理生产器配置时,目标类接口可以省略;
- 后置通知是在目标方法执行之后才执行,它可以得到目标方法的返回值,但是不能改变它;
- 环绕通知有在目标方法执行之前的代码,也有在目标方法执行之后的代码;可以得到目标方法的返回值,且可以改变它。
4.5 Spring实现AOP实例
4.5.1 目标类接口
package com.steven.spring.sysmgr.service;
import java.util.List;
import com.steven.spring.sysmgr.entity.Student;
public interface IStudentService {
public boolean addStudent(Student student);
public boolean delStudent(Integer studentId);
public boolean updateStudent(Student student);
public List<Student> getStudentList();
}
4.5.2 目标类
package com.steven.spring.sysmgr.service.impl;
import java.util.ArrayList;
import java.util.List;
import com.steven.spring.sysmgr.entity.Student;
import com.steven.spring.sysmgr.service.IStudentService;
public class StudentService implements IStudentService{
@Override
public boolean addStudent(Student student) {
System.out.println("执行增加功能");
return true;
}
@Override
public boolean delStudent(Integer studentId) {
//模拟异常,在异常通知中使用
int i = 1 / 0;
System.out.println("执行删除功能");
return true;
}
@Override
public boolean updateStudent(Student student) {
System.out.println("执行修改功能");
return true;
}
@Override
public List<Student> getStudentList() {
System.out.println("执行增加功能");
return new ArrayList<Student>();
}
}
4.5.3 实现通知
a) 前置通知
package com.steven.spring.sysmgr.advice;
import java.lang.reflect.Method;
import org.springframework.aop.MethodBeforeAdvice;
/**
* 前置通知
* @author chenyang
*
*/
public class MyBeforeAdvice implements MethodBeforeAdvice {
@Override
public void before(Method arg0, Object[] arg1, Object arg2) throws Throwable {
// TODO Auto-generated method stub
System.out.println("这是前置通知,应该在方法执行之前打印出来");
}
}
b) 后置通知
package com.steven.spring.sysmgr.advice;
import java.lang.reflect.Method;
import org.springframework.aop.AfterReturningAdvice;
/**
* 后置通知
* @author chenyang
*
*/
public class MyAfterAdvice implements AfterReturningAdvice {
//该方法中的代码就是要织入到目标类方法的切面代码
@Override
public void afterReturning(Object returnValue, Method paraMethod,
Object[] paramArrayOfObject, Object target) throws Throwable {
System.out.println("方法执行的结果:" + returnValue);
System.out.println("这是后置通知,应该在方法执行之后打印出来");
}
}
c) 环绕通知
package com.steven.spring.sysmgr.advice;
import org.aopalliance.intercept.MethodInterceptor;
import org.aopalliance.intercept.MethodInvocation;
/**
* 环绕通知
* @author chenyang
*
*/
public class MyAroundAdvice implements MethodInterceptor {
// 环绕通知可以改变方法的返回值
@Override
public Object invoke(MethodInvocation invocation) throws Throwable {
System.out.println("这是环绕通知的前面执行部分");
Object result = invocation.proceed();
System.out.println("方法的执行结果是:" + result);
System.out.println("这是环绕通知的后面执行部分");
result = false;
return result;
}
}
d) 异常通知
package com.steven.spring.sysmgr.advice;
import org.springframework.aop.ThrowsAdvice;
/**
* 异常通知
* @author chenyang
*
*/
public class MyThrowingAdvice implements ThrowsAdvice {
public void afterThrowing(Exception e){
System.out.println("发生的异常:" + e);
}
}
4.5.4 配置文件:注册目标类、通知、代理生成器
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
<!-- 注册目标类,并描述依赖关系 -->
<bean id="studentService" class="com.steven.spring.sysmgr.service.impl.StudentService"></bean>
<!-- 注册前置通知 -->
<bean id="beforeAdvice" class="com.steven.spring.sysmgr.advice.MyBeforeAdvice"></bean>
<!-- 注册后置通知 -->
<bean id="afterAdvice" class="com.steven.spring.sysmgr.advice.MyAfterAdvice"></bean>
<!-- 注册环绕通知 -->
<bean id="aroundAdvice" class="com.steven.spring.sysmgr.advice.MyAroundAdvice"></bean>
<!-- 注册异常通知 -->
<bean id="throwingAdvice" class="com.steven.spring.sysmgr.advice.MyThrowingAdvice"></bean>
<!-- 注册前置通知的代理生成器,注入目标类接口、目标类、通知 -->
<bean id="myBeforeProxy" class="org.springframework.aop.framework.ProxyFactoryBean">
<property name="target" ref="studentService"/>
<property name="interfaces" value="com.steven.spring.sysmgr.service.IStudentService"/>
<property name="interceptorNames" value="beforeAdvice"/>
</bean>
<!-- 注册后置通知的代理生成器,注入目标类、通知 -->
<bean id="myAfterProxy" class="org.springframework.aop.framework.ProxyFactoryBean">
<property name="target" ref="studentService"/>
<!-- 这里省略了目标类接口的实现类,因为系统能自己找到 -->
<property name="interceptorNames" value="afterAdvice"/>
</bean>
<!-- 注册环绕通知的代理生成器,注入目标类、通知 -->
<bean id="myAroundProxy" class="org.springframework.aop.framework.ProxyFactoryBean">
<property name="target" ref="studentService"/>
<property name="interceptorNames" value="aroundAdvice"/>
</bean>
<!-- 注册异常通知的代理生成器,注入目标类、通知 -->
<bean id="myThrowingProxy" class="org.springframework.aop.framework.ProxyFactoryBean">
<property name="target" ref="studentService"/>
<property name="interceptorNames" value="throwingAdvice"/>
</bean>
</beans>
ps:
教你什么时候用value,什么时候用ref?
ref是引用(指针),value是布尔、string等得值。
用ref还是value,需要看源码中这个name是什么类型。 全选name的引号中的值,alt + / , 若弹出的类的参数是对象,就用ref;若不是,就用value。


4.5.5 测试
package com.steven.spring.sysmgr.test;
import org.junit.Before;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import com.steven.spring.sysmgr.entity.Student;
import com.steven.spring.sysmgr.service.IStudentService;
/**
* 测试各种通知
* @author chenyang
*
*/
public class AdviceTest {
private ApplicationContext ac = null;
@Before
public void init(){
ac = new ClassPathXmlApplicationContext("applicationContext.xml");
}
//测试前置通知,预期目标:将前置通知打印的内容出现在目标方法内容的前面
@Test
public void testBeforeAdvice(){
IStudentService studentService = (IStudentService) ac.getBean("myBeforeProxy");
studentService.addStudent(new Student());
}
//测试后置通知
@Test
public void testAfterAdvice(){
IStudentService studentService = (IStudentService) ac.getBean("myAfterProxy");
studentService.addStudent(new Student());
}
//测试环绕通知
@Test
public void testAroundAdvice(){
IStudentService studentService = (IStudentService) ac.getBean("myAroundProxy");
studentService.addStudent(new Student());
}
//测试异常通知
@Test
public void testThrowingAdvice(){
IStudentService studentService = (IStudentService) ac.getBean("myThrowingProxy");
studentService.delStudent(1);
}
}
上一章:[Spring+SpringMVC+Mybatis]框架学习笔记(三):Spring实现JDBC
下一章:[Spring+SpringMVC+Mybatis]框架学习笔记(五):SpringAOP_顾问
[Spring+SpringMVC+Mybatis]框架学习笔记(四):Spring实现AOP的更多相关文章
- [Spring+SpringMVC+Mybatis]框架学习笔记(六):事务
第7讲 事务 7.1 事务的概念 事务是一系列作为一个逻辑单元来执行的操作集合. 它是数据库维护数据一致性的单位,它讲数据库从一个一致状态,转变为新的另外一个一致状态.说的简单一点就是:如果一组处理步 ...
- 用IntelliJ IDEA 开发Spring+SpringMVC+Mybatis框架 分步搭建四:配置springmvc
在用IntelliJ IDEA 开发Spring+SpringMVC+Mybatis框架 分步搭建三:配置spring并测试的基础上 继续进行springmvc的配置 一:配置完善web.xml文件
- Spring+SpringMVC+MyBatis集成学习笔记【一】
一,首先要清楚,SpringMVC其实就是Spring的一个组件 例如我们知道Spring中有类似于,AOP TX等等类似的组件,所以SpringMVC其实就是Spring的一个组件,是S ...
- 搭建Spring + SpringMVC + Mybatis框架之三(整合Spring、Mybatis和Spring MVC)
整合Spring和SpringMVC 之前已经整合了spring和mybatis,现在在此基础上整合SSM. 项目目录: 思路:SpringMVC的配置文件独立,然后在web.xml中配置整合. (1 ...
- 用IntelliJ IDEA 开发Spring+SpringMVC+Mybatis框架 分步搭建三:配置spring并测试
这一部分的主要目的是 配置spring-service.xml 也就是配置spring 并测试service层 是否配置成功 用IntelliJ IDEA 开发Spring+SpringMVC+M ...
- Spring+SpringMVC+MyBatis深入学习及搭建(四)——MyBatis输入映射与输出映射
转载请注明出处:http://www.cnblogs.com/Joanna-Yan/p/6878529.html 前面有讲到Spring+SpringMVC+MyBatis深入学习及搭建(三)——My ...
- Spring+SpringMVC+MyBatis深入学习及搭建(十四)——SpringMVC和MyBatis整合
转载请注明出处:http://www.cnblogs.com/Joanna-Yan/p/7010363.html 前面讲到:Spring+SpringMVC+MyBatis深入学习及搭建(十三)--S ...
- Spring+SpringMvc+Mybatis框架集成搭建教程
一.背景 最近有很多同学由于没有过SSM(Spring+SpringMvc+Mybatis , 以下简称SSM)框架的搭建的经历,所以在自己搭建SSM框架集成的时候,出现了这样或者那样的问题,很是苦恼 ...
- Spring+SpringMVC+MyBatis深入学习及搭建(二)——MyBatis原始Dao开发和mapper代理开发
转载请注明出处:http://www.cnblogs.com/Joanna-Yan/p/6869133.html 前面有写到Spring+SpringMVC+MyBatis深入学习及搭建(一)——My ...
- Spring+SpringMVC+MyBatis深入学习及搭建(三)——MyBatis全局配置文件解析
转载请注明出处:http://www.cnblogs.com/Joanna-Yan/p/6874672.html 前面有写到Spring+SpringMVC+MyBatis深入学习及搭建(二)——My ...
随机推荐
- 基于kubeasz部署高可用k8s集群
在部署高可用k8s之前,我们先来说一说单master架构和多master架构,以及多master架构中各组件工作逻辑 k8s单master架构 提示:这种单master节点的架构,通常只用于测试环境, ...
- islider.js轮播图
本篇文章地址:https://www.cnblogs.com/Thehorse/p/11601032.html css #iSlider-effect-wrapper { height: 220px; ...
- CF1037G A Game on Strings Sol
有趣题. 首先"分成若干个互不相干的子串"是子游戏的定义,可以用 SG 函数处理. 然而接下来试着打了半个多小时的表,没有找到任何规律. 但是发现 SG 函数的状态转移是很简单的. ...
- 基于.Net开发的数据库导入导出的开源项目
在项目开发过程中,我们经常碰到从数据库导入导出的需求,虽然这样的功能不是很复杂,但是往往我们都会碰到一些问题. 比如导入的Excel格式问题.Excetl中图片导入问题,导出的需求为了方便客户查看,会 ...
- Java的构造方法和标准JavaBean
构造方法 一.构造方法概述: 构造方法也叫做构造器,构造函数,平时叫做构造方法 二.构造方法的作用: 创建对象的时候,由虚拟机自动调用,给成员变量进行初始化(赋值) 三.构造方法的格式: public ...
- ACM中的java的使用;
java大法好,退C保平......开玩笑的: 1.头文件: import java.math.*; // 包含大数类的包 import java.util.*; // 包含输入头的包 2.程序主体, ...
- 2022-09-16:以下go语言代码输出什么?A:101;B:100;C:1;D:2。 package main import ( “fmt“ ) type MS struct {
2022-09-16:以下go语言代码输出什么?A:101:B:100:C:1:D:2. package main import ( "fmt" ) type MS struct ...
- 2022-06-13:golang中,[]byte和结构体如何相互转换?
2022-06-13:golang中,[]byte和结构体如何相互转换? 答案2022-06-13: []byte和结构体的转换的应用场景是数据解析. 代码里有两种方法,一种是内存不共用,另一种是内存 ...
- 认识 CPU 底层原理(2)——逻辑门
本文为B站UP主硬件茶谈制作的系列科普<[硬件科普]带你认识CPU>系列的学习笔记,仅作个人学习记录使用,如有侵权,请联系博主删除 上一篇文章我们从最基本的粒子的角度认识了组成CPU的最基 ...
- 1406, "Data too long for column 'od_seq' at row 1"
问题描述:1406, "Data too long for column 'od_seq' at row 1" 问题分析:录入数据长度超出字段的最大限制 解决方法:增加max_le ...