上一篇分享了一些AOP相关的概念,这一篇继续上一篇分享java动态代理的两种实现方式。
缺陷:JDK的动态代理依靠接口实现,如果类没有实现接口,则不能使用jdk代理,只能使用cglib,但是这也比静态代理好太多。
jdkproxy中包含一个类和一个接口
InvocationHandler接口:
public interface InvocationHandler {
public Object invoke(Object proxy,Method method,Object[] args) throws Throwable;
}
参数说明:
public interface BookFacade {
void addBook();
}
定义该接口的具体实现类
public class BookFacadeImpl implements BookFacade{
@Override
public void addBook() {
System.out.println("--------->【添加图书】<---------");
}
}
实现jdk代理接口InvocationHandler
/**
* created by sunliangliang
* 代理类,jdk代理必须事先InvocationHandler接口
*/
public class BookFacadeProxy implements InvocationHandler{
private Object target;
public Object bind(Object target){
this.target = target;
System.out.println("-----------bind------");
return Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(),this);//this即绑定当前
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
Object result = null;
System.out.println("----------->【任务执行前】<-----------");
result = method.invoke(target,args);
System.out.println("----------->【任务结束】<-----------");
return result;
}
}
测试类
public class TestProxy {
public static void main(String[] args) {
BookFacadeProxy proxy = new BookFacadeProxy();
BookFacade bookProxy = (BookFacade) proxy.bind(new BookFacadeImpl());
bookProxy.addBook();
}
}
那我们看下输出结果
-----------bind------ ----------->【任务执行前】<----------- --------->【添加图书】<--------- ----------->【任务结束】<-----------
总结:由此可以看出使用JDK代理必须实现InvocationHandler接口,
将具体业务类绑定到Proxy这个类上,然后会自动执行invoke()方法。
操作步骤如下:
cglib-nodep-3.2.5.jar ,该jar包中已经包含 asm.jar 的包,若是 cglib-2.x.jar 的版本需要引入 asm.jar ,因为 asm.jar 的版本发生过变更,所以会出现版本冲突的问题。切记jar环境。 public class BookFacadeCglibProxy implements MethodInterceptor{
private Object target;
/**
* 创建代理对象
* @param target
* @return
*/
public Object getInstance(Object target){
this.target = target;
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(target.getClass());
// 回调方法
enhancer.setCallback(this);
//创建代理对象
return enhancer.create();
}
// 回调方法
@Override
public Object intercept(Object obj, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
System.out.println("---------->【开始执行任务】<--------");
methodProxy.invokeSuper(obj,args);
System.out.println("---------->【结束执行任务】<--------");
return null;
}
}
测试类如下:
public class TestCglib {
public static void main(String[] args) {
BookFacadeCglibProxy cglib=new BookFacadeCglibProxy();
BookFacadeImpl bookFacade = (BookFacadeImpl) cglib.getInstance(new BookFacadeImpl());
bookFacade.addBook();
}
}
输出结果如下:
---------->【开始执行任务】<-------- --------->【添加图书】<--------- ---------->【结束执行任务】<--------
总结:
两者最大的区别是jdkproxy需要统一的接口,而cglib不需要。
优点:
优点:
谢谢你请我吃糖果
微信