一 AOP概念
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();
//生成了一个代理类对象proxy
TargetInterface proxy = (TargetInterface)Proxy.newProxyInstance(
//第一个参数被代理类的类加载器
target.getClass().getClassLoader(),
//第二个参数target的所以实现接口
target.getClass().getInterfaces(),
//第三参数是InvocationHandler
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();
}
public interface ClacInterface {
int add(int a ,int b );
}
public class MyClac implements ClacInterface{
@Override
public int add(int a, int b) {
System.out.println("a+b"+"="+(a+b));
return a+b;
}
}
public static void main(String[] args) {
//创建接口的实现类对象,也就是被代理对象
ClacInterface myClac = new MyClac();
//获取被代理实现类对象的Class对象
Class<? extends ClacInterface> clazz = myClac.getClass();
//第一个参数,被代理对象的类加载器
ClassLoader classLoader = clazz.getClassLoader();
//第二个参数,被代理对象实现的所有接口数组
Class<?>[] interfaces = clazz.getInterfaces();
//通过调用Proxy.newProxyInstance的方法来创建一个代理类对象来代理myClac对象
ClacInterface proxyBean = (ClacInterface)Proxy.newProxyInstance(classLoader, interfaces, new InvocationHandler() {
//第三个参数InvocationHandler的实现类,这里用了匿名内部类的方式
@Override
//重写InvocationHandler的invoke方法,他有三个参数可以供我们使用
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//第一个参数proxy 可以通过proxy.getClass().getClassName() 来查看对象,是一个匿名的Proxy的代理类
System.out.println("proxy.getClass().getName() = " + proxy.getClass().getName());
System.out.println("前置增强------");
//第二个参数method 也就是被代理对象myClac对象的方法,可以通过method调用myClac的方法
//第三个参数args 也就是MyClac方法执行时传入的实际参数
//method.invoke的返回值就是myClac方法的返回值
Object invoke = method.invoke(myClac, args);
System.out.println("后置增强------");
return invoke;
}
});
System.out.println("proxyBean.add(1, 3) = " + proxyBean.add(1, 3));
}
- Proxy.newProxyInstance() 方法是生成代理对象的,有三个参数,类加载器 ,目标类的所有接口字节码对象,InvocationHandler
- InvocationHandler : 可以理解为一个回调函数,里面的invoke()是关键,代理对象执行方法就是在执行invoke()方法
- method.invoke(target, args);属于反射,方法的执行,代码如下,作为目前阶段的了解
java反射包中的Proxy类可以调用newProxyInstance创建一个代理类对象. 传入三个参数, 一个是代理类的类加载器,
一个是实现的所有接口数组, 一个调用处理器的实现类.
创建好了代理对象,代理对象就可以执行被代理类实现的接口的方法,在执行方法时,会先去执行调用处理器实现类中的invoke方法,invoke方法就可以对被代理类进行功能增强.
一,先看一下反射的概念:
主要是指程序可以访问,检测和修改它本身状态或行为的一种能力,并能根据自身行为的状态和结果,调整或修改应用所描述行为的状态和相关的语义。
反射是java中一种强大的工具,能够使我们很方便的创建灵活的代码,这些代码可以再运行时装配,无需在组件之间进行源代码链接。但是反射使用不当会成 二,反射机制的作用:
1,反编译:.class-->.java
2,通过反射机制访问java对象的属性,方法,构造方法等;
这样好像更容易理解一些,下边我们具体看怎么实现这些功能。本很高!
public class Target {
public void say1(String name) {
System.out.println("你好"+name);
}
public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InstantiationException, InvocationTargetException {
Class<Target> targetClass = Target.class;
Method method= targetClass.getMethod("say1", String.class);
method.invoke(targetClass.newInstance(),"tom");
}
}
6 cglib代理(了解)
- 使用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>
```
## 代码
```java
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 AOP 相关概念
Target(目标对象):代理的目标对象 98K # Proxy (代理):一个类被 AOP 织入增强后,就产生一个结果代理类
安装了8倍镜的98K #
Joinpoint(连接点):所谓连接点是指那些被拦截到的点。在spring中,这些点指的是方法,因为spring只支持方法类型的连接点。
可以安装8倍境的地方 # Pointcut(切入点):所谓切入点是指我们要对哪些 Joinpoint 进行拦截的定义。
安装了8被镜的地方 # Advice(通知/ 增强):所谓通知是指拦截到 Joinpoint 之后所要做的事情就是通知 8倍镜Aspect(切面):是切入点和通知的结合 # Weaving(织入):是指把增强应用到目标对象来创建新的代理对象的过程。spring采用动态代理织入,而AspectJ采用编译期织入和类装载期织入。
安装的过程