1 什么是 AOP
-
AOP 为 Aspect Oriented Programming 的缩写,意思为面向切面编程,是通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。
-
AOP 是 OOP 的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
2 AOP 的作用及其优势
-
作用:在程序运行期间,在不修改源码的情况下对方法进行功能增强
-
优势:减少重复代码,提高开发效率,并且便于维护
3 AOP 的底层实现
-
AOP 的底层是通过 Spring 提供的的动态代理技术实现的。在运行期间,Spring通过动态代理技术动态的生成代理对象,代理对象方法执行时进行增强功能的介入,在去调用目标对象的方法,从而完成功能的增强。
4 常用的动态代理技术
-
JDK 代理 : 基于接口的动态代理技术
-
cglib 代理:基于父类的动态代理技术
5 JDK代理
/**
* 接口类
*/
public interface TargetInterface {
void say();
}
/**
* 目标类
*/
public class Target implements TargetInterface {
public void say() {
System.out.println("阿乔拉裤子了");
}
}
public static void main( String[] args ) {
final Target target = new Target();
TargetInterface proxy = (TargetInterface)Proxy.newProxyInstance(target.getClass().getClassLoader(),
target.getClass().getInterfaces(),
new InvocationHandler() {
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("他吃多了");
Object invoke = method.invoke(target, args);
System.out.println("拉的很多");
return invoke;
}
});
proxy.say();
}
6 cglib代理
<!--导入spring的context坐标,context依赖aop-->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.0.5.RELEASE</version>
</dependency>
<!-- aspectj的织入 -->
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.8.13</version>
</dependency>
public static void main( String[] args ) {
final Target target = new Target(); //创建目标对象
Enhancer enhancer = new Enhancer(); //创建增强器
enhancer.setSuperclass(Target.class);//设置父类
enhancer.setCallback(new MethodInterceptor() { //设置回调
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
System.out.println("前置代码增强....");
Object invoke = method.invoke(target, objects);
System.out.println("后置代码增强....");
return invoke;
}
});
Target proxy =(Target) enhancer.create();//创建代理对象
proxy.say();
}
7 书写步骤
-
1 导入 AOP 相关坐标
<!--导入spring的context坐标,context依赖aop-->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.0.5.RELEASE</version>
</dependency>
<!-- aspectj的织入 -->
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.8.13</version>
</dependency>
-
2 创建目标接口和目标类
/**
* 接口类
*/
public interface TargetInterface {
void say();
}
/**
* 目标类
*/
public class Target implements TargetInterface {
@Override
public void say() {
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("目标类执行了");
}
}
-
3 创建切面类
public class MyAspect {
//前置增强方法
public void before(){
System.out.println("前置代码增强了");
}
//后置增强方法
public void after(){
System.out.println("后置");
}
//环绕增强
public Object around(ProceedingJoinPoint p){
long start = System.currentTimeMillis();
String className = p.getSignature().getDeclaringTypeName();
String method = p.getSignature().getName();
Object proceed = null;
try {
proceed = p.proceed();
} catch (Throwable throwable) {
throwable.printStackTrace();
}
long end = System.currentTimeMillis();
System.out.println("类名:" + className);
System.out.println("方法名是:" + method);
System.out.println("执行时间长是:"+(end-start));
return proceed;
}
}
-
4 在applicationContext.xml中配置织入关系
-
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:aop="http://www.springframework.org/schema/aop" 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 http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd"> <bean id="myAspect" class="com.xinzhi.aspect.MyAspect" /> <bean id="target" class="com.xinzhi.target.impl.TargetInterfaceImpl" /> <aop:config> <aop:aspect ref="myAspect"> <!-- <aop:before method="before" pointcut="execution(* com.xinzhi.target.impl.*.*(..))"/>--> <!-- <aop:after method="after" pointcut="execution(* com.xinzhi.target.impl.*.*(..))"/>--> <aop:around method="around" pointcut="execution(* com.xinzhi.target.impl.*.*(..))"/> </aop:aspect> </aop:config> </beans>
-
5 测试代码
public static void main( String[] args ) {
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
TargetInterface bean = (TargetInterface)applicationContext.getBean("target");
bean.say();
}