1.1什么是AOP
AOP:Aspect Oriented Programming ⾯向切⾯编程。
1.2AOP面向切面的优势
降低模块之间的耦合度。
使系统更容易扩展。 更好的代码复⽤。
⾮业务代码更加集中,不分散,便于统⼀管理。
业务代码更加简洁存粹,不参杂其他代码的影响。
AOP 是对⾯向对象编程的⼀个补充,在运⾏时,动态地将代码切⼊到类的指定⽅法、指定位置上的编程 思想就是⾯向切⾯编程。将不同⽅法的同⼀个位置抽象成⼀个切⾯对象,对该切⾯对象进⾏编程就是 AOP。
1.3AOP需要添加的依赖
<dependency> <groupId>org.springframework</groupId> <artifactId>spring-aop</artifactId> <version>5.3.9</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-aspects</artifactId> <version>5.3.9</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>5.3.9</version> </dependency>
2.1简述AOP工作运行原理
import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; import java.util.Arrays; public class MyInvocationHandler implements InvocationHandler { private Object object=null; public Object bind(Object object){ this.object=object; return Proxy.newProxyInstance(object.getClass().getClassLoader(),object.getClass().getInterfaces(),this); } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println(method.getName()+"方法的参数"+ Arrays.toString(args)); Object result=method.invoke(this.object,args); System.out.println(method.getName()+"的结果是"+result); return result; } }
以上是动态创建AOP的方法,首先通过bind返回Proxy.newProxyInstance(object.getClass().getClassLoader(),object.getClass.getInterfaces(),this)返回代理对象
接着运用反射的invoke方法,实现面向切面编程
用method.getName()获取方法名
用arg获取属性
2.2动态创建的总结:
以上是通过动态代理实现 AOP 的过程,⽐较复杂,不好理解,Spring 框架对 AOP 进⾏了封装,使⽤ Spring 框架可以⽤⾯向对象的思想来实现 AOP
3.1使用Spring创建AOP
import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.After; import org.aspectj.lang.annotation.AfterReturning; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.springframework.stereotype.Component; import java.util.Arrays; @Aspect @Component public class LoggerAspect { //@before表适方法执行的具体位置和时机 @Before("execution(public int Util.impl.Calimpl.*(..))") public void before(JoinPoint joinPoint){ //获取方法名 String name=joinPoint.getSignature().getName(); //获取参数 String args= Arrays.toString(joinPoint.getArgs()); System.out.println(name+"方法的参数是"+args); } @After("execution(public int Util.impl.Calimpl.*(..))") public void after(JoinPoint joinPoint){ String name=joinPoint.getSignature().getName(); System.out.println(name+"执行完毕"); } @AfterReturning(value = "execution(public int Util.impl.Calimpl.*(..))",returning = "result") public void returning(JoinPoint joinPoint,Object result){ System.out.println("结果是:"+result); } }
注解@Aspect指的是面向切面编程
@Component指的是交给Ioc容器管理
通过Joinpoint.getSignature().getname()获取方法名
然后实现在调用这个方法前,进行运行所需要的日志信息
3.2测试类
import Util.Cal; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; public class Test2 { public static void main(String[] args) { ApplicationContext applicationContext=new ClassPathXmlApplicationContext("spring-aop.xml"); Cal proxy = (Cal) applicationContext.getBean("calimpl"); proxy.add(1,1); } }
还是通过Application 读取Spring.xml 再将getBean 默认小写名字,再调用方法即可
3.3 Spring.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:context="http://www.springframework.org/schema/context" xmlns:aop="http://www.springframework.org/schema/aop" xmlns:p="http://www.springframework.org/schema/p" xsi:schemaLocation="http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.3.xsd http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.3.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.3.xsd "> <!-- ⾃动扫描 --> <context:component-scan base-package="Util"> </context:component-scan> <!-- 是Aspect注解⽣效,为⽬标类⾃动⽣成代理对象 --> <aop:aspectj-autoproxy></aop:aspectj-autoproxy> </beans>