概述
- JDK动态代理:其代理对象必须是某个接口的实现,它是通过在运行期期间创建一个接口的实现类来完成对目标对象的代理。
- CGLIB代理:实现原理类似于JDK动态代理,只是它在运行期间生成的代理对象是针对目标类扩展的子类。CGLIB是高效的代码生成包,底层依靠ASM(开源的Java字节码编辑类库)操作字节码实现。
1 JDK动态代理
1.1 创建代理过程:
- 定义接口
- 实现接口
- 定义代理类,继承InvocationHandler接口
1.2 具体代码
接口:
public interface Person {
void say();
}
实现类:
public class JdkStudent implements Person{
@Override
public void say() {
System.out.println("我是学生...");
}
}
代理类:
public class JdkProxy implements InvocationHandler{
private Object target;
public JdkProxy() {
super();
}
public JdkProxy(Object target) {
this.target = target;
}
/**
该方法负责集中处理动态代理类上的所有方法调用。
第一个参数既是代理类实例,
第二个参数是被调用的方法对象
第三个方法是调用参数。
调用处理器根据这三个参数进行预处理或分派到委托类实例上发射执行
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
System.out.println("------插入前置通知代码-------------");
Object obj = method.invoke(target, args);
System.out.println("------插入后置通知代码-------------");
return obj;
}
}
测试:
public class JdkTest {
public static void main(String[] args) {
//生成$Proxy0的class文件 System.getProperties().put("sun.misc.ProxyGenerator.saveGeneratedFiles", "true");
JdkStudent student = new JdkStudent();
Person p = (Person) Proxy.newProxyInstance(student.getClass().getClassLoader(),
student.getClass().getInterfaces(), new JdkProxy(student));
System.out.println(p);
p.say();
}
}
注:JDK动态代理生成的文件默认在sun/proxy下,如果没有该目录会报Exception in thread “main” java.lang.InternalError: I/O exception saving generated file: java.io.FileNotFoundException: sun\proxy$Proxy0.class (系统找不到指定的路径。)
生成的代理class
package sun.proxy;
import com.gz.design.proxy.jdk.Person;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.lang.reflect.UndeclaredThrowableException;
public final class $Proxy0
extends Proxy
implements Person
{
private static Method m3;
private static Method m1;
private static Method m0;
private static Method m2;
public $Proxy0(InvocationHandler paramInvocationHandler)
{
super(paramInvocationHandler);
}
public final void say()
{
try
{
this.h.invoke(this, m3, null);
return;
}
catch (Error|RuntimeException localError)
{
throw localError;
}
catch (Throwable localThrowable)
{
throw new UndeclaredThrowableException(localThrowable);
}
}
public final boolean equals(Object paramObject)
{
try
{
return ((Boolean)this.h.invoke(this, m1, new Object[] { paramObject })).booleanValue();
}
catch (Error|RuntimeException localError)
{
throw localError;
}
catch (Throwable localThrowable)
{
throw new UndeclaredThrowableException(localThrowable);
}
}
public final int hashCode()
{
try
{
return ((Integer)this.h.invoke(this, m0, null)).intValue();
}
catch (Error|RuntimeException localError)
{
throw localError;
}
catch (Throwable localThrowable)
{
throw new UndeclaredThrowableException(localThrowable);
}
}
public final String toString()
{
try
{
return (String)this.h.invoke(this, m2, null);
}
catch (Error|RuntimeException localError)
{
throw localError;
}
catch (Throwable localThrowable)
{
throw new UndeclaredThrowableException(localThrowable);
}
}
static
{
try
{
m3 = Class.forName("com.gz.design.proxy.jdk.Person").getMethod("say", new Class[0]);
m1 = Class.forName("java.lang.Object").getMethod("equals", new Class[] { Class.forName("java.lang.Object") });
m0 = Class.forName("java.lang.Object").getMethod("hashCode", new Class[0]);
m2 = Class.forName("java.lang.Object").getMethod("toString", new Class[0]);
return;
}
catch (NoSuchMethodException localNoSuchMethodException)
{
throw new NoSuchMethodError(localNoSuchMethodException.getMessage());
}
catch (ClassNotFoundException localClassNotFoundException)
{
throw new NoClassDefFoundError(localClassNotFoundException.getMessage());
}
}
}
2 cglib动态代理
1.1 创建代理过程:
- 创建Enhancer对象
- 用Enhancer对象设置代理类的父类(被代理类)
- 创建回调对象(回调类实现 MethodInterceptor 接口)
- 用Enhancer对象设置回调对象
- 用Enhancer对象创建代理对象
1.2 具体代码
目标类:
`
public class Student{
public void say() {
System.out.println(“我是学生…”);
test();
}
public void test() {
System.out.println("test");
}
}
**代理类:**
public class CglibProxy implements MethodInterceptor{
/**
* 重写方法拦截在方法前和方法后加入业务
* Object obj为目标对象
* Method method为目标方法
* Object[] params 为参数,
* MethodProxy proxy CGlib方法代理对象
*/
@Override
public Object intercept(Object obj, Method method, Object[] args,
MethodProxy methodProxy) throws Throwable {
System.out.println("===============");
return methodProxy.invokeSuper(obj, args);
}
}
**测试类:**
public class CglibProxyTest {
public static void main(String[] args) throws NoSuchFieldException, SecurityException, IllegalArgumentException, IllegalAccessException {
// 生成class类的路径
System.setProperty(DebuggingClassWriter.DEBUG_LOCATION_PROPERTY, "E://tmp");
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(Student.class);
enhancer.setCallback(new CglibProxy());
Student student = (Student) enhancer.create();
student.say();
}
}
**生成的代理class**
package com.gz.design.proxy.cglib;
import java.lang.reflect.Method;
import net.sf.cglib.core.ReflectUtils;
import net.sf.cglib.proxy.Callback;
import net.sf.cglib.proxy.Factory;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
public class Student
extends Student
implements Factory
{
private boolean CGLIB THREAD_CALLBACKS;
private static final Callback[] CGLIB CALLBACK_0;
private static final Method CGLIB 0 say Proxy;
private static final Object[] CGLIB finalize Method;
private static final MethodProxy CGLIB 1 equals Method;
private static final MethodProxy CGLIB 2 toString Method;
private static final MethodProxy CGLIB 3 hashCode Method;
private static final MethodProxy CGLIB 4 clone Method;
private static final MethodProxy CGLIB 5$Proxy;
static void CGLIB
THREAD_CALLBACKS = new ThreadLocal();
CGLIB
say
Method = tmp50_47[0];
CGLIB
0
say
finalize
Method = tmp143_140[0];
CGLIB
1
finalize
equals
Method = tmp163_143[1];
CGLIB
2
equals
toString
Method = tmp183_163[2];
CGLIB
3
toString
hashCode
Method = tmp203_183[3];
CGLIB
4
hashCode
clone
Method = tmp223_203[4];
CGLIB
5
clone$5”);
tmp223_203;
return;
}
final void CGLIB
0()
{
super.say();
}
public final void say()
{
MethodInterceptor tmp4_1 = this.CGLIB
CALLBACK_0 != null) {
return;
}
super.say();
}
final void CGLIB
1()
throws Throwable
{
super.finalize();
}
protected final void finalize()
throws Throwable
{
MethodInterceptor tmp4_1 = this.CGLIB
CALLBACK_0 != null) {
return;
}
super.finalize();
}
final boolean CGLIB
2(Object paramObject)
{
return super.equals(paramObject);
}
public final boolean equals(Object paramObject)
{
MethodInterceptor tmp4_1 = this.CGLIB
CALLBACK_0;
if (tmp17_14 != null)
{
Object tmp41_36 = tmp17_14.intercept(this, CGLIB
2
equals
Proxy);
tmp41_36;
return tmp41_36 == null ? false : ((Boolean)tmp41_36).booleanValue();
}
return super.equals(paramObject);
}
final String CGLIB
3()
{
return super.toString();
}
public final String toString()
{
MethodInterceptor tmp4_1 = this.CGLIB
CALLBACK_0;
if (tmp17_14 != null) {
return (String)tmp17_14.intercept(this, CGLIB
3
emptyArgs, CGLIB
3$Proxy);
}
return super.toString();
}
final int CGLIB
4()
{
return super.hashCode();
}
public final int hashCode()
{
MethodInterceptor tmp4_1 = this.CGLIB
CALLBACK_0;
if (tmp17_14 != null)
{
Object tmp36_31 = tmp17_14.intercept(this, CGLIB
4
emptyArgs, CGLIB
4$Proxy);
tmp36_31;
return tmp36_31 == null ? 0 : ((Number)tmp36_31).intValue();
}
return super.hashCode();
}
final Object CGLIB
5()
throws CloneNotSupportedException
{
return super.clone();
}
protected final Object clone()
throws CloneNotSupportedException
{
MethodInterceptor tmp4_1 = this.CGLIB
CALLBACK_0;
if (tmp17_14 != null) {
return tmp17_14.intercept(this, CGLIB
5
emptyArgs, CGLIB
5$Proxy);
}
return super.clone();
}
/* Error */
public static MethodProxy CGLIB
finalize
Proxy Lnet/sf/cglib/proxy/MethodProxy;
// 79: areturn
// 80: ldc -122
// 82: invokevirtual 132 java/lang/Object:equals (Ljava/lang/Object;)Z
// 85: ifeq +56 -> 141
// 88: getstatic 46 com/gz/design/proxy/cglib/Student
// 103: areturn
// 104: ldc -118
// 106: invokevirtual 132 java/lang/Object:equals (Ljava/lang/Object;)Z
// 109: ifeq +32 -> 141
// 112: getstatic 77 com/gz/design/proxy/cglib/Student
// 127: areturn
// 128: ldc -114
// 130: invokevirtual 132 java/lang/Object:equals (Ljava/lang/Object;)Z
// 133: ifeq +8 -> 141
// 136: getstatic 107 com/gz/design/proxy/cglib/Student
// 139: areturn
// 140: pop
// 141: aconst_null
// 142: areturn
}
public Student
{
CGLIB$BIND_CALLBACKS(this);
}
public static void CGLIB
THREAD_CALLBACKS.set(paramArrayOfCallback);
}
public static void CGLIB
STATIC_CALLBACKS = paramArrayOfCallback;
}
private static final void CGLIB
BOUND)
{
local4b734e7.CGLIB
THREAD_CALLBACKS.get();
if (tmp23_20 == null)
{
tmp23_20;
CGLIB
CALLBACK_0 = (tmp31_28 == null ? tmp31_28 : (MethodInterceptor)((Callback[])tmp23_20)[0]);
}
}
public Object newInstance(Callback[] paramArrayOfCallback)
{
CGLIB
SET_THREAD_CALLBACKS(null);
return new 4b734e7();
}
public Object newInstance(Callback paramCallback)
{
CGLIB
SET_THREAD_CALLBACKS(null);
return new 4b734e7();
}
/* Error */
public Object newInstance(Class[] arg1, Object[] arg2, Callback[] arg3)
{
// Byte code:
// 0: aload_3
// 1: invokestatic 206 com/gz/design/proxy/cglib/Student
// 53: areturn
}
public Callback getCallback(int paramInt)
{
CGLIB$BIND_CALLBACKS(this);
switch (paramInt)
{
case 0:
break;
}
return null;
}
public void setCallback(int paramInt, Callback paramCallback)
{
switch (paramInt)
{
case 0:
this.CGLIB$CALLBACK_0 = ((MethodInterceptor)paramCallback);
break;
}
}
public Callback[] getCallbacks()
{
CGLIB
CALLBACK_0 };
}
public void setCallbacks(Callback[] paramArrayOfCallback)
{
this.CGLIB$CALLBACK_0 = ((MethodInterceptor)paramArrayOfCallback[0]);
}
static {}
}
“`