动态代理的的两种实现方式

一、动态代理简介

动态代理实际上是JVM在运行期动态创建class字节码并加载的过程。

优势:在不修改源码的情况下,对目标方法进行相应的增强。

作用:完成程序功能之间的松耦合。

二、动态代理的两种实现

JDK代理:基于接口的动态代理技术(缺点,目标对象必须有接口,如果没有接口,则无法完成动态代理的实现)

cglib代理:基于父类的动态代理技术

两者的区别如图所示:
image.png

1. 基于JDK的实现

目标接口类:

1
2
3
4
5
csharp复制代码public interface TargetInterface {
public void save();

public void print(String str);
}

目标类:

1
2
3
4
5
6
7
8
9
10
11
typescript复制代码public class Target implements TargetInterface{

public void save() {
System.out.println("save running...");
}

public void print(String str) {
System.out.println(str);
}

}

增强类:

1
2
3
4
5
6
7
8
9
csharp复制代码public class Advice {
public void before() {
System.out.println("前置增强");
}

public void after() {
System.out.println("后置增强");
}
}

测试类:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
java复制代码import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

public class ProxyTest {

public static void main(String[] args) {

//目标对象
final Target target = new Target();

//增强对象
final Advice advice = new Advice();

TargetInterface proxyInstance = (TargetInterface)Proxy.newProxyInstance(
target.getClass().getClassLoader(), //目标对象类加载器
target.getClass().getInterfaces(), //目标对象相同的接口字节码对象数组
new InvocationHandler() {
//调用代理对象的任何方法,实质执行的都是invoke方法
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable{
advice.before(); //前置增强
Object invoke = method.invoke(target, args); //执行目标方法
advice.after(); //后置增强
System.out.println();
return invoke;
}
});

//代理对象的方法测试
proxyInstance.save();

proxyInstance.print("JDK动态代理");
}

}

运行截图:

image.png

2. 基于cglib的实现

需要导入Jar包,如果是maven项目,则在pom.xml文件加入如下配置:

1
2
3
4
5
xml复制代码<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>4.2.4.RELEASE</version>
</dependency>

目标类:

1
2
3
4
5
6
7
8
9
csharp复制代码public class Target {
public void save() {
System.out.println("save running...");
}

public void print(String str) {
System.out.println(str);
}
}

增强类:

1
2
3
4
5
6
7
8
9
10
11
csharp复制代码public class Advice {

public void before() {
System.out.println("前置增强");
}

public void after() {
System.out.println("后置增强");
}

}

测试类:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
java复制代码import java.lang.reflect.Method;

import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;

public class ProxyTest {

public static void main(String[] args) {
final Target target = new Target();
final Advice advice = new Advice();

//返回值就是动态生成的代理对象,基于cglib
//创建增强器
Enhancer enhancer = new Enhancer();

//设置父类(目标)
enhancer.setSuperclass(Target.class);

//设置回调
enhancer.setCallback(new MethodInterceptor() {
public Object intercept(Object o, Method method, Object[] obj, MethodProxy methodProxy) throws Throwable{
advice.before();
Object invoke = method.invoke(target, obj);
advice.after();
System.out.println();
return invoke;
}
});

//创建代理对象
Target proxy = (Target)enhancer.create();

//测试代理方法
proxy.save();
proxy.print("基于cglib实现动态规划");

}

}

运行截图:

image.png

三、为什么要有基于cglib的实现

使用JDK动态代理实现时,最大限制是被增强对象必须实现接口,并且增强的方法只能是接口中声明的方法。但在实际的项目中,可能总是存在对不实现业务接口的对象进行增强的需求,这时JDK动态代理将无能为力。

本文转载自: 掘金

开发者博客 – 和开发相关的 这里全都有

0%