AOP示例(三续)

使用JDK创建代理有一个限制,即它只能为接口创建代理,这一点我们从Proxy的接口方法newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h)就看得很清楚,第三个入参interfaces就是为代理实例指定的实现接口。虽然,面向接口的编程被很多很有影响力人(包括Rod Johnson)的推崇,但在实际开发中,开发者也遇到了很多困惑:难道对一个简单业务表的操作真的需要创建5个类(领域对象类、Dao接口,Dao实现类,Service接口和Service实现类)吗?对于这一问题,我们还是留待大家进一步讨论。现在的问题是:对于没有通过接口定义业务方法的类,如何动态创建代理实例呢?JDK的代理技术显然已经黔驴技穷,CGLib作为一个替代者,填补了这个空缺

下面得演示还是居于上面一直在使用得例子

1.新建CglibProxy代理类

2.修改MethodAdvice类的invoke方法,添加private MethodProxy porxy;这样一个属性,及其set方法

3.修改Test类,改成使用CglibProxy代理

  1. /**
  2. *
  3. */
  4. packagecom.advice;
  5. importjava.lang.reflect.Method;
  6. importnet.sf.cglib.proxy.Enhancer;
  7. importnet.sf.cglib.proxy.MethodInterceptor;
  8. importnet.sf.cglib.proxy.MethodProxy;
  9. /**
  10. *@authorzjb
  11. *
  12. */
  13. publicclassCglibProxyimplementsMethodInterceptor{
  14. privateMethodAdvicemethodAdvice;
  15. privateEnhancerenhancer=newEnhancer();
  16. publicObjectgetBean(StringclassName){
  17. try{
  18. enhancer.setSuperclass(Class.forName(className));
  19. enhancer.setCallback(this);
  20. returnenhancer.create();
  21. }catch(Exceptione){
  22. e.printStackTrace();
  23. }
  24. returnnull;
  25. }
  26. publicMethodAdvicegetMethodAdvice(){
  27. returnmethodAdvice;
  28. }
  29. publicvoidsetMethodAdvice(MethodAdvicemethodAdvice){
  30. this.methodAdvice=methodAdvice;
  31. }
  32. @Override
  33. publicObjectintercept(Objectobject,Methodmethod,Object[]args,
  34. MethodProxyporxy)throwsThrowable{
  35. methodAdvice.setArgs(args);
  36. methodAdvice.setMethod(method);
  37. methodAdvice.setPos(-1);
  38. methodAdvice.setDone(false);
  39. methodAdvice.setTarget(object);
  40. methodAdvice.setPorxy(porxy);
  41. returnmethodAdvice.proceed();
  42. }
  43. }
修改以后的MethodAdvice类的invoke方法

  1. privateObjectinvoke(){
  2. done=true;
  3. try{
  4. if(porxy!=null){
  5. returnporxy.invokeSuper(target,args);
  6. }else{
  7. returnmethod.invoke(target,args);
  8. }
  9. }catch(Exceptione){
  10. e.printStackTrace();
  11. }catch(Throwablee){
  12. e.printStackTrace();
  13. }
  14. returnnull;
  15. }

  1. packagecom.myadvice.test;
  2. importcom.advice.CglibProxy;
  3. importcom.advice.DynaProxy;
  4. importcom.advice.MethodAdvice;
  5. importcom.test.BookBiz;
  6. importcom.test.BookBizImpl;
  7. publicclassTest{
  8. /**
  9. *@paramargs
  10. */
  11. publicstaticvoidmain(String[]args){
  12. //DynaProxyproxy=newDynaProxy();
  13. CglibProxyproxy=newCglibProxy();
  14. MethodAdvicemethodAdvice=newMethodAdvice();
  15. methodAdvice.AddAdvice(newObject[]{newMyAdvice()});
  16. proxy.setMethodAdvice(methodAdvice);
  17. //BookBizbookBiz=(BookBiz)proxy.getBean(BookBizImpl.class.getName());
  18. BookBizImplbookBiz=(BookBizImpl)proxy.getBean(BookBizImpl.class.getName());
  19. bookBiz.buy("张三","Spring深入潜出",50);
  20. bookBiz.comment("李四","《恐怖世界》一点都不恐怖,很好看!");
  21. bookBiz.comment("张三","《Spring深入潜出》还是写得不错的!");
  22. }
  23. }

下面是运行结果:

[环绕通知]
前通知,调用的方法:buy,参数:[张三, Spring深入潜出, 50.0]
业务方法buy开始执行
·张三购买图书:Spring深入潜出
·张三增加积分:5
·向物流系统下发货单
业务方法buy结束
后通知,调用的方法:buy,参数:[张三, Spring深入潜出, 50.0]
[环绕通知]
屏蔽李四所有的评论
[环绕通知]
前通知,调用的方法:comment,参数:[张三, 《Spring深入潜出》还是写得不错的!]
业务方法comment开始执行
·张三发表书评《Spring深入潜出》还是写得不错的!
业务方法comment结束
后通知,调用的方法:comment,参数:[张三, 《Spring深入潜出》还是写得不错的!]

通过学习AOP,俺们现在回去想想Spring的数据库事务自动管理是怎么实现的,就比较清晰了,应该使用了环绕通知,在proceed之前开始一个事务,然后在提交事务,就这么简单而已。(^_^,大功告成,关于aop打完收工)

你可能感兴趣的:(DAO,spring,jdk,AOP,物流)