日期:2014-05-17  浏览次数:20844 次

问题:Spring基于@AspectJ参数绑定的原理
  Spring 2.0提供了基于@AspectJ和Schema的AOP配置,有一个问题一直让我很郁闷,那就是增强方法是如何绑定到目标类连接点方法的入参的,Spring通过argNames成员指定入参名来绑定连接点的入参,在Spring的文档中说argNames是为了明确绑定入参名,请看下面的代码:
	@Before("target(com.baobaotao.NaiveWaiter) && args(name,num,..)",argsName="name,num")
	public void bindJoinPointParams(int num,String name){
	   System.out.println("----bindJoinPointParams()----");
	   System.out.println("name:"+name);
	   System.out.println("num:"+num);
	   System.out.println("----bindJoinPointParams()----");
	}

   在切点表达式中,通过args(name,num,..)绑定了两个连接点方法的参数,一个匹配的连接点方法如下:
package com.baobaotao;
public class NaiveWaiter implements Waiter {
   public void smile(String name,int num){
	System.out.println("NaiveWaiter:smile to  "+name+ num+"times...");
   }	
}

   上面的切点,将name和num分别绑定到增强方法的name和num参数中.
   我的问题是,既然args(name,num,..)已经指定了参数名,为什么还需要通过argsName再次声明参数名呢?Spring文档告诉我们:argsName是明确绑定的参数名,说是反射机制无法获取入参名。关键是这种明确的指定有两个问题:

   1)首先,这里指定的信息和切点表达式中args(name,num)中指定的信息是重复的;
   2)其次,我们无法通过反射机制获取目标类连接点smile(String name,int num)参数名,在切面定义中明确指定入参名又有什么意思呢?这好比说,张三和歪毛已经消了名,现在我们确以张三歪毛为线索去查找两人,这有用吗?


   实际上,我们将切点定义处的argNames成员去除,我发现效果是一样的,它照样可以成功绑定。这个问题的本质是既然Java反射机制无法获取入参名,Spring到底是用了何种神秘的机制,获取方法入参名,因为增强方法和目标连接点方法是通过名称关联绑定的,而不是通过类型绑定的。解决了这个问题后,另一个问题才是为什么需要提供argName这个搞怪的成员做重复性的声明工作。
1 楼 shaucle 2007-01-29  
InvocationHandler 中有args,也可以打印啊
public Object invoke(Object proxy, Method method, Object[] args)