云存储做网站seo网站排名优化快速排
web/
2025/10/8 4:45:19/
文章来源:
云存储做网站,seo网站排名优化快速排,南京模板网站开发,域名估价网站给Advice传递参数 Advice除了可以接收JoinPoint#xff08;非Around Advice#xff09;或ProceedingJoinPoint#xff08;Around Advice#xff09;参数外#xff0c;还可以直接接收与切入点方法执行有关的对象#xff0c;比如切入点方法参数、切入点目标对象#xff08…给Advice传递参数 Advice除了可以接收JoinPoint非Around Advice或ProceedingJoinPointAround Advice参数外还可以直接接收与切入点方法执行有关的对象比如切入点方法参数、切入点目标对象target、切入点代理对象this等。 5.1 获取切入点方法参数 假设我们现在有一个id为userService的bean中定义了一个findById(int id)方法我们希望定义一个Advice来拦截这个方法并且把findById()的参数作为Advice处理方法的参数即每次调用findById()传递的参数都将传递到Advice处理方法那么我们可以如下这样定义。 1 Before(valuebean(userService) execution(* findById(java.lang.Integer)) args(id), argNamesid)
2 public void beforeWithParam(JoinPoint joinPoint, Integer id) {
3 System.out.println(this.getClass().getName() ID is : id);
4 } 上面这种定义是非常精确的定义我们通过表达式“bean(userService) execution(* findById(java.lang.Integer))”就已经明确的指定了我们需要拦截的是id或name为userService的findById(Integer)方法后面又加了一个args(id)是干什么用的呢它的作用跟findById(Integer)是类似的它表示我们的切入点方法必须只接收一个参数而且这个参数的类型是和当前定义的Advice处理方法的参数id是相同类型的在上面的示例中其实就是要求是Integer类型的另外它还有一个非常重要的作用通过这种指定后对应的Advice处理方法在执行时将接收到与之对应的切入点方法参数的值。在上面的示例中笔者特意给Advice处理方法加了一个JoinPoint参数是为了说明JoinPoint、ProceedingJoinPoint参数是可以直接定义在Advice方法的第一个参数并且是可以与其它接收的参数共存的。其实如果我们不只是需要拦截findById(Integer)方法而是需要拦截id为userService的bean中所有接收一个int/Integer参数的方法那么我们可以把上面的配置简化为如下这样。 1 Before(valuebean(userService) args(id), argNamesid)
2 public void beforeWithParam2(int id) {
3 System.out.println(this.getClass().getName() ID is : id);
4 } 如果我们需要拦截的方法可能是有多个参数的但我们只关注第一个参数那我们可以把表达式调整为如下这样只关注第一个参数为int/Integer类型的并且在Advice方法中接收这个方法参数进行相应的处理。 1 Before(valuebean(userService) args(id,..), argNamesid)
2 public void beforeWithParam2(int id) {
3 System.out.println(this.getClass().getName() ID is : id);
4 } 5.2 argNames参数 我们可以看到在上述例子中我们都指定了Before的argNames属性的值为id那么这个argNames属性有什么作用呢argNames属性是用于指定在表达式中应用的参数名与Advice方法参数是如何对应的argNames中指定的参数名必须与表达式中的一致可以与Advice方法参数名不一致当表达式中使用了多个参数时argNames中需要指定多个参数多个参数之间以英文逗号分隔这些参数的顺序必须与对应的Advice方法定义的参数顺序是一致的。比如下面这个示例中我们在Pointcut表达式中使用了name和sex两个参数我们的Advice处理方法接收两个参数分别是sex1和name1我们希望Pointcut表达式中的name参数是对应的Advice处理方法的第二个参数即name1希望Pointcut表达式中的sex参数是对应的Advice处理方法的第一个参数即sex1那么我们在指定Before注解的argNames参数时必须定义name和sex参数与Advice处理方法参数的关系且顺序要求与对应的处理方法的参数顺序一致即哪个参数是需要与Advice处理方法的第一个参数匹配则把哪个参数放第一位与第二个参数匹配的则放第二位在我们的这个示例中就应该是sex放第一位name放第二位。 1 Before(valuebean(userService) args(name, sex), argNamessex, name)
2 public void beforeWithParam3(int sex1, String name1) {
3 System.out.println(sex is : sex1);
4 System.out.println(name is : name1);
5 } Before注解的argNames参数不是必须的它只有在我们编译的字节码中不含DEBUG信息或Pointcut表达式中使用的参数名与Advice处理方法的参数名不一致时才需要。所以在编译的字节码中包含DEBUG信息且Advice参数名与Pointcut表达式中使用的参数名一致时我们完全可以把argNames参数省略。如果表达式里面使用了多个参数那么这些参数在表达式中的顺序可以与Advice方法对应参数的顺序不一致例如下面这个样子。 1 Before(valuebean(userService) args(id))
2 public void beforeWithParam2(int id) {
3 System.out.println(this.getClass().getName() ID is : id);
4 } 5.3 获取this对象 this对象就是Spring生成的bean的那个代理对象。如下示例就是Advice方法接收this对象我们给Advice方法指定一个需要拦截的this对象类型的参数然后在表达式中使用this类型的表达式定义表达式中定义的对应类型指定为Advice方法参数。 1 Before(this(userService))
2 public void beforeWithParam4(IUserService userService) {
3 //this对象应该是一个代理对象
4 System.out.println(this.getClass().getName()传递this对
5 象 userService.getClass());
6 } 5.4 混合使用 我们的Advice方法可以同时接收多个目标方法参数与此同时它也可以接收this等对象即它们是可以混合使用的。下面这个示例中我们就同时接收了this对象和目标方法int/Interger类型的参数。 1 Before(this(userService) args(id))
2 public void beforeWithParam5(IUserService userService, int id) {
3 System.out.println(this.getClass().getName() id
4 userService.getClass());
5 } 5.5 获取target对象 获取target对象也比较简单只需要把表达式改为target类型的表达式即可。 1 Before(target(userService))
2 public void beforeWithParam6(IUserService userService) {
3 System.out.println(this.getClass().getName()传递
4 target对象 userService.getClass());
5 } 5.6 获取注解对象 当我们的Pointcut表达式类型是通过注解匹配时我们也可以在Advice处理方法中获取匹配的注解对象如下面这个示例其它如使用target等是类似的。 1 Before(annotation(annotation))
2 public void beforeWithParam7(MyAnnotation annotation) {
3 System.out.println(this.getClass().getName()传递标
4 注在方法上的annotation annotation.annotationType().getName());
5 } 5.7 泛型参数 有的时候我们的Advice方法需要接收的切入点方法参数定义的不是具体的类型而是一个泛型这种情况下怎么办呢可能你会想那我就把对应的Advice方法参数定义为Object类型就好了反正所有的类型都可以转换为Object类型。对的这样是没有错的但是说如果你只想拦截某种具体类型的参数调用时就可以不用把Advice方法参数类型定义为Object了这样还得在方法体里面进行判断我们可以直接把Advice方法参数类型定义为我们想拦截的方法参数类型。比如我们有下面这样一个使用了泛型的方法定义我们希望只有在调用testParam方法时传递的参数类型是Integer类型时才进行拦截。 1 public T void testParam(T param); 那这个时候我们就可以把我们的Advice的表达式定义为如下这样前者精确定义接收方法名为testParam返回类型为void后者精确定义方法参数为一个Integer类型的参数其实前者也可以定义为“execution(void testParam(Integer))”。看到这你可能会想为什么不直接把表达式定义为“execution(void testParam(param))”呢因为execution是不支持Advice方法参数绑定的基本上支持Advice参数绑定的就只有this、target、args以及对应的注解形式加annotation。 1 Before(execution(void testParam(..)) args(param))
2 public void beforeWithParam8(Integer param) {
3 System.out.println(pointcut expression[args(param)]--------------param:
4 param);
5 } 以上就是常用的传递参数给Advice处理方法的方式有一些示例可能没有讲到比如target这种这些其实都是类似的。包括上面我们都是以Before这种Advice来讲的其实其它的Advice在接收参数的时候也是类似的。 转载于:https://www.cnblogs.com/fnlingnzb-learner/p/10716049.html
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/88876.shtml
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!