java实现动态代理方法浅析
一些Java项目中在mybatis与spring整合中有MapperScannerConfigurer的使用,该类通过反向代理自动生成基于接口的动态代理类。
有鉴于此,本文浅析了java的动态代理。
本文使用动态代理模拟处理事务的拦截器。
接口:
publicinterfaceUserService{
publicvoidaddUser();
publicvoidremoveUser();
publicvoidsearchUser();
}
实现类:
publicclassUserServiceImplimplementsUserService{
publicvoidaddUser(){
System.out.println("adduser");
}
publicvoidremoveUser(){
System.out.println("removeuser");
}
publicvoidsearchUser(){
System.out.println("searchuser");
}
}
java动态代理的实现有2种方式
1.jdk自带的动态代理
使用jdk自带的动态代理需要了解InvocationHandler接口和Proxy类,他们都是在java.lang.reflect包下。
InvocationHandler介绍:
InvocationHandler是代理实例的调用处理程序实现的接口。
每个代理实例都具有一个关联的InvocationHandler。对代理实例调用方法时,这个方法会调用InvocationHandler的invoke方法。
Proxy介绍:
Proxy提供静态方法用于创建动态代理类和实例。
实例(模拟AOP处理事务):
publicclassTransactionInterceptorimplementsInvocationHandler{
privateObjecttarget;
publicvoidsetTarget(Objecttarget){
this.target=target;
}
@Override
publicObjectinvoke(Objectproxy,Methodmethod,Object[]args)throwsThrowable{
System.out.println("startTransaction");
method.invoke(target,args);
System.out.println("endTransaction");
returnnull;
}
}
测试代码:
publicclassTestDynamicProxy{
@Test
publicvoidtestJDK(){
TransactionInterceptortransactionInterceptor=newTransactionInterceptor();
UserServiceuserService=newUserServiceImpl();
transactionInterceptor.setTarget(userService);
UserServiceuserServiceProxy=
(UserService)Proxy.newProxyInstance(
userService.getClass().getClassLoader(),
userService.getClass().getInterfaces(),
transactionInterceptor);
userServiceProxy.addUser();
}
}
测试结果:
startTransaction adduser endTransaction
很明显,我们通过userServiceProxy这个代理类进行方法调用的时候,会在方法调用前后进行事务的开启和关闭。
2.第三方库cglib
CGLIB是一个功能强大的,高性能、高质量的代码生成库,用于在运行期扩展Java类和实现Java接口。
它与JDK的动态代理的之间最大的区别就是:
JDK动态代理是针对接口的,而cglib是针对类来实现代理的,cglib的原理是对指定的目标类生成一个子类,并覆盖其中方法实现增强,但因为采用的是继承,所以不能对final修饰的类进行代理。
实例代码如下:
publicclassUserServiceCallBackimplementsMethodInterceptor{
@Override
publicObjectintercept(Objecto,Methodmethod,Object[]args,MethodProxymethodProxy)throwsThrowable{
System.out.println("startTransactionbycglib");
methodProxy.invokeSuper(o,args);
System.out.println("endTransactionbycglib");
returnnull;
}
}
测试代码:
publicclassTestDynamicProxy{
@Test
publicvoidtestCGLIB(){
Enhancerenhancer=newEnhancer();
enhancer.setSuperclass(UserServiceImpl.class);
enhancer.setCallback(newUserServiceCallBack());
UserServiceImplproxy=(UserServiceImpl)enhancer.create();
proxy.addUser();
}
}
测试结果:
startTransactionbycglib adduser endTransactionbycglib
感兴趣的读者可以实际测试一下本文实例,相信会有很大的收获。