1)在目标类源代码不改变的情况下,增加功能。
2)减少代码的重复
3)专注业务逻辑代码
4)解耦合,让你的业务功能和日志,事务非业务功能分离。
代理设计模式详细笔记 :代理模式详细笔记,想学好Spring的AOP思想先理解代理模式
java设计模式之代理设计模式笔记详细内容:
-
代理模式是结构型模式其中的一种
-
现在开发中存在的问题
-
什么是代理模式,为什么需要使用代理模式
-
静态代理及实现
-
什么是动态代理
-
JDK 动态代理和cglib动态代理的使用及区别
-
三种动态代理的对比及优缺点
-
代理模式的使用场景
=============================================================================
Spring 的 AOP 实现底层就是对动态代理的代码进行了封装 ,封装后我们只需要对需要关注的部分进行代码编写,并通过配置的方式完成指定目标的方法增强。
在正式讲解 AOP 的操作之前,我们必须理解 AOP 的相关术语,常用的术语如下:
1、切面(Aspect)
一个切面就是一个代理对象= 要为那些类生成代理+要添加的额外功能是那些
2、切入点(pointcut):将来要为那些类生成代理对象
3、通知/ 增强(advice):就是要添加的额外功能
- 生活案例:给面包之间涂果酱
注意:切面(Aspect)=切入点(pointcut)+通知点(advice,额外功能)
4、AOP中的(面向切面编程)通知
前置通知:目标方法之前的额外功能 MethodBeforeAdvice
环绕通知:它是spring框架为我们提供的一种可以在代码中手动控制增强方法何时执行的方式 MethodInterceptor(Interceptor [ɪntəˈsɛptə]拦截器)
后置通知:目标方法之后的额外功能 AfterReturningAdvice(returning[rɪˈtɜːrnɪŋ])
异常通知:执行异常的额外功能 ThrowsAdvice
最终通知:一定执行的额外功能
5、Target(目标对象):代理的目标对象
6、代理(Proxy[ˈprɑːksi]):一个类被AOP注入增强后,就产生一个结果代理类
7、连接点(joinPoint):所谓连接点是指那些被拦截到的点。在spring中,这些点指的是方法(额外功能),因为spring只支持方法类型的连接点
===========================================================================
1、五种通知类
AOP 切面=切入点+通知
-
前置通知:MethodBeforeAdvice
-
后置通知:AfterReturnAdvice
-
环绕通知:MethodInterceptor
-
异常通知:ThrowsAdvice(throw [θroʊz])
-
最终通知
- 通知的配置语法:
<aop : 通知类型 method=“切面类中方法名” pointcut=“切点表达式"> </aop:通知类型>
- 切点表达式的抽取
当多个增强的切点表达式相同时,可以将切点表达式进行抽取,在增强中使用 pointcut-ref 属性代替 pointcut 属性来引用抽取后的切点表达式。
aop:config
<aop:aspect ref=“myAspect”>
<aop:pointcut id=“myPointcut” expression=“execution(* com.tjcu.aop..(…))”/>
<aop:before method=“before” pointcut-ref=“myPointcut”></aop:before>
</aop:aspect>
</aop:config>
- aop织入的配置
aop:config
<aop:aspect ref=“切面类”>
<aop:before method=“通知方法名称” pointcut=“切点表达式"></aop:before>
</aop:aspect>
</aop:config>
- 切点表达式的写法:
execution([修饰符] 返回值类型 包名.类名.方法名(参数))
2、AOP的开发步骤
1、开发目标类
2、开发通知类,确定额外功能
3、管理通知类
4、配置切入点 确定要为那些类添加额外功能
5、将目标类和切面类的对象创建权交给 spring
6、组装切面 切入点+通知(额外功能)
3、AOP编程所需要的依赖
引入依赖
spring-aop
spring-expression
spring-aspects
================================================================================
1、目标接口类
public interface CityService {
public void login();
public void add(String name);
}
2、目标实现类(核心功能)
/**
-
@author 王恒杰
-
@Description:目标对象
*/
public class CityServiceImpl implements CityService{
@Override
public void login() {
//前置通知:System.out.println(“嘻嘻哈哈”);
//执行核心的业务逻辑 调用Dao
System.out.println(“登录调用Dao”);
}
@Override
public void add(String name) {
//前置通知:System.out.println(“嘻嘻哈哈”);
//执行核心的业务逻辑 调用Dao
System.out.println(“添加调用Dao”);
}
}
3、前置通知(额外功能)动态代理代码
/**
-
@author 王恒杰
-
@Description: 通知类。额外功能
*/
public class MyBeforeAdvice implements MethodBeforeAdvice {
/**
-
额外功能书写的方法
-
参数1:代理对象当前调用的方法
-
参数2:当前代理对象调用的方法的参数
-
参数3:目标对象(被代理的对象)
-
@param method
-
@param objects
-
@param o
-
@throws Throwable
*/
@Override
public void before(Method method, Object[] objects, Object o) throws Throwable {
System.out.println(“嘻嘻哈哈”);
}
}
4、将目标类和切面类的对象创建权交给 spring
5、在 spring.xml 中配置织入关系(前置功能)
aop:config
<aop:pointcut id=“pc1” expression=“execution(* before.CityServiceImpl.*(…))”/>
<aop:advisor advice-ref=“myBeforeAdvice” pointcut-ref=“pc1”></aop:advisor>
</aop:config>
6、测试代码
@Test
public void testMethodBeforeAdvice() {
//启动工厂
ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext(“before/spring.xml”);
//获取组件 目标对象就是代理对象
CityService cityService = (CityService) ctx.getBean(“cityService”);
//目标对象就是代理对象 class com.sun.proxy.$Proxy4
System.out.println(cityService.getClass());
//调用方法,通过代理类调用目标类
cityService.add(“123”);
}
注意: 获取组件时,目标对象就是代理对象
===================================================================================
1、dao层接口
public interface StudentDao {
/**
-
登录
-
@param name
*/
public void login(String name);
/**
-
分页
-
@param name
-
@return
*/
public String pageShow(String name);
}
2、dao层实现类
public class StudentDaoImpl implements StudentDao{
@Override
public void login(String name) {
//循环10000次
for (int i = 0; i < 10000; i++) {
}
System.out.println(“数据库实现登录”);
}
@Override
public String pageShow(String name) {
//循环10000次
for (int i = 0; i < 10000; i++) {
}
System.out.println(“数据库实现分页”);
return name;
}
}
3、目标接口类
public interface StudentService {
/**
-
登录
-
@param name
*/
public void login(String name);
/**
-
分页
-
@param name
-
@return
*/
public String pageShow(String name);
}
4、目标实现类
public class StudentServiceImpl implements StudentService{
private StudentDao studentDao;
public void setStudentDao(StudentDao studentDao) {
this.studentDao = studentDao;
}
@Override
public void login(String name) {
System.out.println(“登录日志”);
studentDao.login(name);
}
@Override
public String pageShow(String name) {
System.out.println(“分页日志”);
String s = studentDao.pageShow(name);
return s;