1. 静态代理与动态代理区别

静态代理:需要编写批量代理类
动态代理:不需要编写批量代理类
2.静态代理
静态代理结构如下:
编写测试接口:
public interface TestInteface {
/**
* 测试方法
*/
String runNow(String param);
}//end
编写测试类:
public class TestProxy implements TestInteface {
public String runNow(String param) {
System.out.println("param:" + param);
return "result:" + param;
}
}//end
编写静态代理类:
public class StaticProxy implements TestInteface {
private TestInteface testInteface;
public StaticProxy(TestInteface testInteface) {
this.testInteface = testInteface;
}
@Override
public String runNow(String param) {
System.out.println("静态代理开始");
String result = testInteface.runNow(param);
System.out.println("静态代理结束");
return result;
}
}//end
编写main方法:
public static void main(String[] args) {
TestInteface testInteface = new TestProxy();
StaticProxy staticProxy = new StaticProxy(testInteface);
String s = staticProxy.runNow("这是静态代理");
System.out.println(s);
}
运行结果如下:
3.动态代理
动态代理分为JDK动态代理、CGLIB动态代理。
JDK动态代理结构如下:
编写通用代理类:
public class JdkProxy implements InvocationHandler {
/**
* 目标类
*/
private Object target;
public JdkProxy(Object target) {
this.target = target;
}
/*
* @Param: proxy 动态代理的实例
* @Param: method 目标类的方法
* @Param: args 目标类方法的参数
* @Return: 目标类方法的执行结果
* @Description:
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("JDK动态代理开始");
Object result = method.invoke(target, args);
System.out.println("JDK动态代理结束");
return result;
}
}//end
编写main方法:
public static void main(String[] args) {
try {
TestInteface testInteface = new TestProxy();
JdkProxy jdkProxy = new JdkProxy(testInteface);
TestInteface testProxy_ = (TestInteface) Proxy.newProxyInstance(testInteface.getClass().getClassLoader(), testInteface.getClass().getInterfaces(), jdkProxy);
String r = testProxy_.runNow("这是JDK动态代理");
System.out.println(r);
} catch (Exception e) {
System.out.println(e);
}
}
运行结果如下:
CGLIB动态代理结构如下:
由于JDK动态代理是JDK原生,CGLIB动态代理是第三方,所以需要引入如下jar包:
compile 'cglib:cglib:3.3.0' compile group: 'org..ant', name: 'ant', version: '1.10.3' compile group: 'org.ow2.asm', name: 'asm', version: '7.1'
编写通用代理类:
public class CGLIBProxy implements MethodInterceptor {
/**
* @Param: obj 目标对象
* @Param: method 目标对象方法
* @Param: args 目标对象方法参数
* @Param: proxy 代理类方法
* @Return:
*/
@Override
public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
System.out.println("CGLIB动态代理开始");
Object result = proxy.invokeSuper(obj, args);
System.out.println("CGLIB动态代理结束");
return result;
}
}//end
编写main方法:
public static void main(String[] args) {
CGLIBProxy cglibProxy = new CGLIBProxy();
//强化剂实例化
Enhancer enhancer = new Enhancer();
//强化目标类
enhancer.setSuperclass(TestProxy.class);
//使用通用代理类进行强化
enhancer.setCallback(cglibProxy);
//创建目标对象
TestProxy o = (TestProxy) enhancer.create();
String s = o.runNow("这是CGLIB动态代理");
System.out.println(s);
}
运行结果如下:
4.总结
静态代理和动态代理都需要编写代理类,不同的是动态代理不需要批量编写代理类,只是需要编写一个通用代理类就行了。JDK动态代理中实现类必须实现接口,CGLIB则没有此限制。
代理的使用场景:鉴权、打印方法前后日志、减少代码的侵入性。
胜象大百科







