前边章节已经介绍了声明通知,但如果想获取被被通知方法参数并传递给通知方法,该如何实现呢?接下来我们将介绍两种获取通知参数的方式。
-
使用joinpoint获取:spring aop提供使用org.aspectj.lang.joinpoint类型获取连接点数据,任何通知方法的第一个参数都可以是joinpoint(环绕通知是proceedingjoinpoint,joinpoint子类),当然第一个参数位置也可以是joinpoint.staticpart类型,这个只返回连接点的静态部分。
1) joinpoint:提供访问当前被通知方法的目标对象、代理对象、方法参数等数据:
java代码:
package org.aspectj.lang;
import org.aspectj.lang.reflect.sourcelocation;
public interface joinpoint {
string tostring(); //连接点所在位置的相关信息
string toshortstring(); //连接点所在位置的简短相关信息
string tolongstring(); //连接点所在位置的全部相关信息
object getthis(); //返回aop代理对象
object gettarget(); //返回目标对象
object[] getargs(); //返回被通知方法参数列表
signature getsignature(); //返回当前连接点签名
sourcelocation getsourcelocation();//返回连接点方法所在类文件中的位置
string getkind(); //连接点类型
staticpart getstaticpart(); //返回连接点静态部分
}
2)proceedingjoinpoint:用于环绕通知,使用proceed()方法来执行目标方法:
java代码:
public interface proceedingjoinpoint extends joinpoint {
public object proceed() throws throwable;
public object proceed(object[] args) throws throwable;
}
3) joinpoint.staticpart:提供访问连接点的静态部分,如被通知方法签名、连接点类型等:
java代码:
public interface staticpart {
signature getsignature(); //返回当前连接点签名
string getkind(); //连接点类型
int getid(); //唯一标识
string tostring(); //连接点所在位置的相关信息
string toshortstring(); //连接点所在位置的简短相关信息
string tolongstring(); //连接点所在位置的全部相关信息
}
使用如下方式在通知方法上声明,必须是在第一个参数,然后使用jp.getargs()就能获取到被通知方法参数:
java代码:
@before(value="execution(* saybefore(*))")
public void before(joinpoint jp) {}
@before(value="execution(* saybefore(*))")
public void before(joinpoint.staticpart jp) {}
-
自动获取:通过切入点表达式可以将相应的参数自动传递给通知方法,例如前边章节讲过的返回值和异常是如何传递给通知方法的。
在spring aop中,除了execution和bean指示符不能传递参数给通知方法,其他指示符都可以将匹配的相应参数或对象自动传递给通知方法。
java代码:
@before(value="execution(* test(*)) && args(param)", argnames="param")
public void before1(string param) {
system.out.println("===param:" param);
}
切入点表达式execution(* test(*)) && args(param) :
1)首先execution(* test(*))匹配任何方法名为test,且有一个任何类型的参数;
2)args(param)将首先查找通知方法上同名的参数,并在方法执行时(运行时)匹配传入的参数是使用该同名参数类型,即java.lang.string;如果匹配将把该被通知参数传递给通知方法上同名参数。
其他指示符(除了execution和bean指示符)都可以使用这种方式进行参数绑定。
在此有一个问题,即前边提到的类似于【3.1.2构造器注入】中的参数名注入限制:在class文件中没生成变量调试信息是获取不到方法参数名字的。
所以我们可以使用策略来确定参数名:
1、如果我们通过“argnames”属性指定了参数名,那么就是要我们指定的;
java代码:
@before(value=" args(param)", argnames="param") //明确指定了
public void before1(string param) {
system.out.println("===param:" param);
}
2、如果第一个参数类型是joinpoint、proceedingjoinpoint或joinpoint.staticpart类型,应该从“argnames”属性省略掉该参数名(可选,写上也对),这些类型对象会自动传入的,但必须作为第一个参数;
java代码:
@before(value=" args(param)", argnames="param") //明确指定了
public void before1(joinpoint jp, string param) {
system.out.println("===param:" param);
}
3、如果“class文件中含有变量调试信息”将使用这些方法签名中的参数名来确定参数名;
java代码:
@before(value=" args(param)") //不需要argnames了
public void before1(joinpoint jp, string param) {
system.out.println("===param:" param);
}
4、如果没有“class文件中含有变量调试信息”,将尝试自己的参数匹配算法,如果发现参数绑定有二义性将抛出ambiguousbindingexception异常;对于只有一个绑定变量的切入点表达式,而通知方法只接受一个参数,说明绑定参数是明确的,从而能配对成功。
java代码:
@before(value=" args(param)")
public void before1(joinpoint jp, string param) {
system.out.println("===param:" param);
}
5、以上策略失败将抛出illegalargumentexception。
接下来让我们示例一下组合情况吧:
java代码:
@before(args(param) && target(bean) && @annotation(secure)",
argnames="jp,param,bean,secure")
public void before5(joinpoint jp, string param,
ipointcutservice pointcutservice, secure secure) {
……
}
该示例的执行步骤如图6-5所示。
图6-5 参数自动获取流程
除了上边介绍的普通方式,也可以对使用命名切入点自动获取参数:
java代码:
@pointcut(value="args(param)", argnames="param")
private void pointcut1(string param){}
@pointcut(value="@annotation(secure)", argnames="secure")
private void pointcut2(secure secure){}
@before(value = "pointcut1(param) && pointcut2(secure)",
argnames="param, secure")
public void before6(joinpoint jp, string param, secure secure) {
……
}
自此给通知传递参数已经介绍完了,示例代码在cn.javass.spring.chapter6.parametertest文件中。
原创内容,转载请注明出处【】
相关推荐
【第六章】 aop 之 6.6 通知参数 ——跟我学spring3 【第六章】 aop 之 6.7 通知顺序 ——跟我学spring3 【第六章】 aop 之 6.8 切面实例化模型 ——跟我学spring3 【第六章】 aop 之 6.9 代理机制 ——跟我学...
1.25 【第六章】 aop 之 6.6 通知参数 ——跟我学spring3 . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .274 1.26 【第六章】 aop 之 6.7 通知顺序 ——跟我学spring3 . . . . . . . . . . ...
spring概述2.1 ioc基础2.2 ioc 容器基本原理2.3 ioc的配置使用——跟我学spring33.1 di的配置使用3.2 循环依赖3.3 更多di的知识 3.4 bean的作用域 4.1 基础知识4.2 内置resource实现4.3 访问resource4.4 resource...
spring中aop中标签加载通知,spring中aop中标签加载通知spring中aop中标签加载通知spring中aop中标签加载通知,spring中aop中标签加载通知
死磕spring之aop篇 - spring aop两种代理对象的拦截处理(csdn)————程序
spring aop spring aop spring aop spring aop spring aop spring aop spring aop spring aop spring aop
spring-aop-1.1.1.jar spring-aop-1.2.6.jar spring-aop-1.2.9.jar spring-aop-2.0.2.jar spring-aop-2.0.6.jar spring-aop-2.0.7.jar spring-aop-2.0.8.jar spring-aop-2.0.jar spring-aop-2.5.1.jar spring-aop-...
详细讲解了 ioc di aop jdbc mvc 等等spring知识,有很高的学习价值
开发工具 spring-aop-4.3.6.release开发工具 spring-aop-4.3.6.release开发工具 spring-aop-4.3.6.release开发工具 spring-aop-4.3.6.release开发工具 spring-aop-4.3.6.release开发工具 spring-aop-4.3.6.release...
spring_aop_学习小结 spring_aop_学习小结 spring_aop_学习小结 spring_aop_学习小结
springaop xml方式配置通知
aop修改方法的参数,如何截取指定方法的参数并修改其参数值
跟我学spring3,包括spring 对象管理,aop,mvc,resource等,非常详细!
spring框架的关键组件之一是面向方面编程(aop)框架。 面向方面的编程需要将程序逻辑分解成不同的部分。 此教程将通过简单实用的方法来学习spring框架提供的aop/面向方面编程。
spring之aop(动态代理),包括jdk动态代理和cglib动态代理
spring aop
springaop与spring定时器
spring 使用aspectj 实现 aop之前置通知小例子,实际跑过,验证可信。
aop是spring的核心技术。通过aop 实现上层的许多高级功能
配套教材,里面的代码测试过都是可运行的,配套学习很方便