spring-AOP(面向切面编程)-xml方式配置
AOP是针对面向对象编程的一种补充,有时使用面向对象不能很好完成一些额外的功能业务时,可以采用AOP来进行补充。
AOP术语:
切面(Aspect)
切面是用于编写切面逻辑的一个类,这个类很类似于JDK动态代理中的回调处理器或者cglib中的方法拦截器,主要就是将需要增强目标对象的功能代码编写在这个类中,而这些功能增强的代码就是切面逻辑。
切入点(Pointcut)
切入点类似一个切入的坐标,目的就是要找到目标对象的哪些方法。
通常切入点都是以一种表达式的形式来描述
通知/增强(Advice)
通知就是切面中具体的增强逻辑,总共分为五种:
1)前置通知(在目标方法调用之前执行)
2)后置通知(在目标方法正确返回之后执行)
3)环绕通知(在目标方法调用前后执行)
4)异常通知(当目标方法抛出异常时执行,并且不会执行后置通知)
5)最终通知(不管目标方法有无异常都会执行)
连接点(Joinpoint)
目标对象的方法就称之为连接点,一个切入点可以对应目标对象的的多个连接点。
代理(Proxy)
在运行时动态创建的对象,称之为代理对象,负责调用目标对象的方法,并执行增强功能
目标(Target)
被代理的对象就是目标对象
织入(Weaver)
将切面中的通知应用到目标对象上并且产生代理的过程称之为织入。
因此通常描述为“将通知织入到具体的目标”。
项目结构:

代码示例:
applicationContext.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: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/aop
http://www.springframework.org/schema/aop/spring-aop.xsd"> <!-- 装备UserServiceImpl -->
<bean id="userService" class="edu.nf.ch11.service.impl.UserServiceImpl"/> <!-- 装配自定义的切面-->
<bean id="userServiceAspect" class="edu.nf.ch11.service.aspect.UserServiceAspect"/> <!-- 配置AOP, proxy-target-class用于设置是否强制使用cglib进行动态代理
true表示强制使用,false则表示spring会根据目标对象有无实现接口来决定
使用jdk动态代理还是cglib代理。默认值就是false-->
<aop:config>
<!-- 配置切入点,id属性给切入点定义一个唯一标识,expression用于编写切入点表达式-->
<!-- execution表达式的使用:切入范围是在方法级别-->
<!-- 表达式语法[访问修饰符] 返回值类型 [完整类名].方法名(参数)-->
<!-- *号表示通配所有,方法的参数可以使用".."来代表任意个数和类型的参数-->
<aop:pointcut id="myCut" expression="execution(* edu.nf.ch11.service.*.*(..))"/> <!-- within表达式的使用:切入范围是在类级别-->
<!--<aop:pointcut id="myCut" expression="within(edu.nf.ch11.service.impl.*)"/>--> <!-- 引用上面装配的切面类的id -->
<aop:aspect ref="userServiceAspect">
<!-- 配置具体的通知方法,method指定通知的方法名
pointcut-ref引用上面定的切入点的id,也可以通过pointcut来编写相应的切入点表达式 -->
<!-- 前置通知 -->
<aop:before method="before" pointcut-ref="myCut"/>
<!-- 环绕通知-->
<aop:around method="around" pointcut-ref="myCut"/>
<!-- 后置通知,returning指定后置通知的参数名,用于获取目标方法的返回值-->
<aop:after-returning method="afterReturn" pointcut-ref="myCut" returning="returnVal"/>
<!-- 异常通知,如果要获取目标方法抛出的异常对象,需要指定throwing属性,value对应异常通知的参数名-->
<aop:after-throwing method="throwAdvice" pointcut-ref="myCut" throwing="e"/>
<!-- 最终通知-->
<aop:after method="after" pointcut-ref="myCut"/>
</aop:aspect> </aop:config> </beans>
切面(增强)类:
package edu.nf.ch11.service.aspect; import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.Signature;
import org.aspectj.lang.reflect.MethodSignature; import javax.xml.crypto.dsig.SignatureMethod; /**
* @author wangl
* @date 2018/10/23
* 编写一个UserService的切面
* 在切面中编写的方法都称之为通知或者是增强
* 在AOP当中,通知有五种类型
* 1. 前置通知
* 2. 后置通知
* 3. 环绕通知
* 4. 最终通知
* 5. 异常通知
* 在一个切面中,任何一种类型的通知都可以定义多个,当有多个通知存在的时候
* 就会形成一个通知栈
*/
public class UserServiceAspect { /**
* 前置通知
* 在执行目标方法之前执行,拦截目标方法的参数
* 可以通过JoinPoint获得目标方法的参数信息
*/
public void before(JoinPoint joinPoint){
System.out.println("前置通知...");
//通过连接点获得目标对象
//joinPoint.getTarget();
//获取目标方法的参数信息
Object[] params = joinPoint.getArgs();
for (Object param : params) {
System.out.println(param);
}
} /**
* 后置通知
* 在目标方法执行完并且return之后执行
*/
public void afterReturn(String returnVal){
System.out.println("后置通知..."+returnVal);
} /**
* 环绕通知
* @param pjp 连接点处理器,由它负责调用目标对象的具体方法
*
*/
public Object around(ProceedingJoinPoint pjp) throws Throwable {
System.out.println("环绕通知前...");
//调用目标对象的方法
Object returnVal = pjp.proceed();
//获取目标方法的参数
System.out.println(pjp.getArgs()[0]);
//通过MethodSignature获取连接点方法信息
MethodSignature ms = (MethodSignature)pjp.getSignature();
//获取正在调用的目标方法
System.out.println(ms.getMethod().getName());
//获取目标方法的返回类型
System.out.println(ms.getReturnType());
System.out.println("环绕通知后...");
return returnVal;
} /**
* 异常通知
* @param e 获取目标方法抛出的异常对象
*/
public void throwAdvice(Throwable e){
System.out.println("异常通知..."+e.getMessage());
} /**
* 最终通知
*/
public void after(){
System.out.println("最终通知...");
}
}
UserService接口:
package edu.nf.ch11.service; /**
* @author wangl
* @date 2018/10/23
*/
public interface UserService { /**
* 添加用户
*/
void addUser(); /**
* 删除用户
* @param uid
*/
void deleteUser(String uid); /**
* 查询用户
* @param uid
* @return
*/
String getUserNameById(String uid);
}
UserServiceImpl实现类:
package edu.nf.ch11.service.impl; import edu.nf.ch11.service.UserService; /**
* @author wangl
* @date 2018/10/23
* 目标对象(被代理的对象)
* Spring的AOP中需要代理的所有目标对象都应该归纳在ioc容器中管理
*/
public class UserServiceImpl implements UserService { @Override
public void addUser() {
System.out.println("保存用户信息");
} @Override
public void deleteUser(String uid) {
System.out.println("删除用户,ID:" + uid);
} @Override
public String getUserNameById(String uid) {
System.out.println("根据ID查询用户名");
//System.out.println(10/0);
return "user1";
}
}
程序测试类:
package edu.nf.ch11.test; import edu.nf.ch11.service.UserService;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext; /**
* @author wangl
* @date 2018/10/23
*/
public class UserServiceAspectTest { @Test
public void testAddUser(){
ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
//这里从容器中获取的对象是一个代理对象
UserService service = context.getBean("userService", UserService.class);
//service.addUser();
//System.out.println("----------------------");
//service.deleteUser("10001");
//System.out.println("----------------------");
service.getUserNameById("10001");
}
}
运行结果

spring-AOP(面向切面编程)-xml方式配置的更多相关文章
- 详细解读 Spring AOP 面向切面编程(二)
本文是<详细解读 Spring AOP 面向切面编程(一)>的续集. 在上篇中,我们从写死代码,到使用代理:从编程式 Spring AOP 到声明式 Spring AOP.一切都朝着简单实 ...
- 浅谈Spring AOP 面向切面编程 最通俗易懂的画图理解AOP、AOP通知执行顺序~
简介 我们都知道,Spring 框架作为后端主流框架之一,最有特点的三部分就是IOC控制反转.依赖注入.以及AOP切面.当然AOP作为一个Spring 的重要组成模块,当然IOC是不依赖于Spring ...
- spring AOP面向切面编程学习笔记
一.面向切面编程简介: 在调用某些类的方法时,要在方法执行前或后进行预处理或后处理:预处理或后处理的操作被封装在另一个类中.如图中,UserService类在执行addUser()或updateUse ...
- 【Spring系列】Spring AOP面向切面编程
前言 接上一篇文章,在上午中使用了切面做防重复控制,本文着重介绍切面AOP. 在开发中,有一些功能行为是通用的,比如.日志管理.安全和事务,它们有一个共同点就是分布于应用中的多处,这种功能被称为横切关 ...
- 从源码入手,一文带你读懂Spring AOP面向切面编程
之前<零基础带你看Spring源码--IOC控制反转>详细讲了Spring容器的初始化和加载的原理,后面<你真的完全了解Java动态代理吗?看这篇就够了>介绍了下JDK的动态代 ...
- Spring AOP面向切面编程详解
前言 AOP即面向切面编程,是一种编程思想,OOP的延续.在程序开发中主要用来解决一些系统层面上的问题,比如日志,事务,权限等等.在阅读本文前希望您已经对Spring有一定的了解 注:在能对代码进行添 ...
- 详细解读 Spring AOP 面向切面编程(一)
又是一个周末, 今天我要和大家分享的是 AOP(Aspect-Oriented Programming)这个东西,名字与 OOP 仅差一个字母,其实它是对 OOP 编程方式的一种补充,并非是取而代之. ...
- Spring Aop面向切面编程&&自动注入
1.面向切面编程 在程序原有纵向执行流程中,针对某一个或某一些方法添加通知,形成横切面的过程叫做面向切面编程 2.常用概念 原有功能:切点,pointcut 前置通知:在切点之前执行的功能,befor ...
- Spring AOP 面向切面编程入门
什么是AOP AOP(Aspect Oriented Programming),即面向切面编程.众所周知,OOP(面向对象编程)通过的是继承.封装和多态等概念来建立一种对象层次结构,用于模拟公共行为的 ...
随机推荐
- java中String类为什么不可变?
在面试中经常遇到这样的问题:1.什么是不可变对象.不可变对象有什么好处.在什么情景下使用它,或者更具体一点,java的String类为什么要设置成不可变类型? 1.不可变对象,顾名思义就是创建后的对象 ...
- leetcode — minimum-window-substring
import java.util.HashMap; import java.util.Map; /** * * Source : https://oj.leetcode.com/problems/mi ...
- 你真的懂redis的数据结构了吗?redis内部数据结构和外部数据结构揭秘
Redis有哪些数据结构? 字符串String.字典Hash.列表List.集合Set.有序集合SortedSet. 很多人面试时都遇到过这种场景吧? 其实除了上面的几种常见数据结构,还需要加上数据结 ...
- 谈谈 JAVA 的对象序列化
所谓的『JAVA 对象序列化』就是指,将一个 JAVA 对象所描述的所有内容以文件 IO 的方式写入二进制文件的一个过程.关于序列化,主要涉及两个流,ObjectInputStream 和 Objec ...
- koa2入门使用总结
koa2的介绍 Koa 是一个新的 web 框架,由 Express 幕后的原班人马打造, 致力于成为 web 应用和 API 开发领域中的一个更小.更富有表现力.更健壮的基石. 通过利用 async ...
- Java反射,注解,以及动态代理
Java反射,注解,以及动态代理 基础 最近在准备实习面试,被学长问到了Java反射,注解和动态代理的内容,发现有点自己有点懵,这几天查了很多资料,就来说下自己的理解吧[如有错误,望指正] Java ...
- mysql服务器架构
mysql是最广泛使用的开源数据库之一,作为后端开发人员,或多或少都会和mysql打交道,本篇文章会从sql查询语句的执行过程,来介绍mysql的服务器架构, 查询的过程大致分为从客户端到服务器,在服 ...
- 解读经典《C#高级编程》第七版 Page50-68.核心C#.Chapter2
前言 本篇讲述Main方法,控制台,注释,预处理指令,编程规范等.这些概念比较琐碎,为避免长篇大论,主要以列举要点的方式来说明. 01 Main方法 Main方法并不是所有应用类型的入口方法,它只是控 ...
- [转]认识JWT
本文转自:https://www.cnblogs.com/cjsblog/p/9277677.html 1. JSON Web Token是什么 JSON Web Token (JWT)是一个开放标准 ...
- [转]【Angular4】基础(六):HTTP模块
本文转自:https://blog.csdn.net/u013451157/article/details/79519719 版权声明:本文为博主原创文章,未经博主允许不得转载. https://bl ...