Spring AOP功能最基本的技術(shù)要點為動態(tài)代理吧恃。
當(dāng)下Java主要有兩種動態(tài)代理方式
- 基于接口的JDK動態(tài)代理
- 基于類CGLib動態(tài)代理
JDK動態(tài)代理
JDK動態(tài)代理是在運行時根據(jù)類的接口生成新的實現(xiàn)類玖姑,讓新的實現(xiàn)類對已有對象進行代理浇雹。
首先聲明兩個接口
public interface Machine {
public void start();
}
public interface Car extends Machine {
public void drive();
}
實現(xiàn)類
public class Bus implements Car {
@Override
public void drive() {
System.out.println("老司機開車?yán)玻?);
}
@Override
public void start() {
System.out.println("開始:炕酢返劲!");
}
}
代理處理類:
public class JdkProxy implements InvocationHandler {
private Object obj;
@SuppressWarnings( "unchecked" )
public <T> T getProxy( T obj ) {
this.obj = obj;
return (T) Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj.getClass().getInterfaces(), this);
}
@Override
public Object invoke( Object proxy, Method method, Object[] args ) throws Throwable {
System.out.println("動態(tài)代理開始病苗!");
Object reVal = method.invoke(obj, args);
System.out.println("動態(tài)代理結(jié)束惩嘉!");
return reVal;
}
}
Proxy.newProxyInstance是使用當(dāng)前類的接口生成代理對象。
InvocationHandler 為代理的處理接口乱灵。
Main函數(shù)
public static void main( String[] args ) {
Car car = new Bus();
JdkProxy jdkProxy = new JdkProxy();
Car proxyCar = jdkProxy.getProxy(car);
proxyCar.start();
proxyCar.drive();
}
結(jié)果
CGLib動態(tài)代理
CGLib是生成當(dāng)前類的子類忆某,讓子類對父類進行代理。
引入Jar包
<dependency>
<groupId>cglib</groupId>
<artifactId>cglib</artifactId>
<version>3.2.4</version>
</dependency>
被代理的類
public class Student {
public void sayHello() {
System.out.println("hello");
}
}
代理處理及代理對象生成
public class StudentProxy implements MethodInterceptor {
private Enhancer enhancer;
public <T> T getProxy( Class<T> clazz ) {
enhancer = new Enhancer();
enhancer.setSuperclass(clazz);
enhancer.setCallback(this);
return (T) enhancer.create();
}
@Override
public Object intercept( Object object, Method method, Object[] args, MethodProxy proxy ) throws Throwable {
System.out.println("before");
proxy.invokeSuper(object, args);
System.out.println("after");
return null;
}
}
Main函數(shù):
public static void main( String[] args ) {
StudentProxy proxy = new StudentProxy();
Student student = proxy.getProxy(Student.class);
student.sayHello();
}
結(jié)果:
結(jié)果
簡單測試兩種代理方式的性能:
/**
* @author : Zak
* @date : 2017年1月17日 上午11:52:23
* @version : 2017年1月17日 Zak 首次創(chuàng)建
*/
public class Main {
public static void main( String[] args ) {
Bus bus = new Bus();
JdkProxy jdkProxy = new JdkProxy();
CglibProxy cglibProxy = new CglibProxy();
long jdkproxyInitTime = 0, cglibproxyinitTime = 0, jdkInvokeTime = 0, cglibInvokeTime = 0, startTime = 0;
for( int i = 0; i < 1000001; i++ ) {
System.out.println(i);
switch(i % 2) {
case 0:// JDKProxy
startTime = System.currentTimeMillis();
Car car = jdkProxy.getProxy(bus);
jdkproxyInitTime += System.currentTimeMillis() - startTime;
startTime = System.currentTimeMillis();
car.drive();
jdkInvokeTime += System.currentTimeMillis() - startTime;
break;
case 1:// CGLib
startTime = System.currentTimeMillis();
Car car1 = cglibProxy.getProxy(Bus.class);
cglibproxyinitTime += System.currentTimeMillis() - startTime;
startTime = System.currentTimeMillis();
car1.drive();
cglibInvokeTime += System.currentTimeMillis() - startTime;
}
}
System.out.println("~~~~~~~~~~~~~~~~~~~~~Result~~~~~~~~~~~~~~~~~~~~~~~~~~");
System.out.println("JDKInitTtime:" + jdkproxyInitTime);
System.out.println("jdkInvokeTime:" + jdkInvokeTime);
System.out.println("jdkInvokeTime:" + cglibproxyinitTime);
System.out.println("cglibInvokeTime:" + cglibInvokeTime);
}
}
測試結(jié)果