解决@Around对静态方法不生效的问题

场景:

在处理定时任务时,由于这几个方法都是静态方法,在aop的切面中使用@Around注解,进行监控方法调用是否有异常。

发现aop没有生效。

代码如下:

/*切面类*/
@Aspect
@Component
public class RetryAop {
 private static Logger logger = LoggerFactory.getLogger(RetryAop.class);
 @Around(value = "@annotation(TechlogRetry)")
 public Object monitorAround(ProceedingJoinPoint pjp) throws Throwable {
 }
}
/*对应的注解,在需要的方法上进行标注*/
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface TechlogRetry {
 long waitMsec() default 0;
 Class[] retryThrowable() default {};
}
/*对应的测试方法*/
@Component
@EnableScheduling
public class TimerWork {
 // 成员方法可以成功调用
 @Scheduled(cron = "*/5 * * * * ?")
 @TechlogRetry(waitMsec = 3000,retryThrowable = Exception.class)
 public void work1() {
  System.out.println("定时方法1.。。。。。。");
  throw new NullPointerException("抛出异常");
 }
  // 静态方法不会进行重试
	@Scheduled(cron = "*/5 * * * * ?")
 @TechlogRetry(waitMsec = 3000,retryThrowable = Exception.class)
 public static void work2() {
  System.out.println("定时方法2.。。。。。。");
  throw new NullPointerException("抛出异常");
 }
 // 通过这种方式生效
 @Scheduled(cron = "*/5 * * * * ?")
 @TechlogRetry(waitMsec = 3000,retryThrowable = Exception.class)
 public void work4() {
  System.out.println("这是work4.。。。");
  TimerWork.work2();// 调用work2时,注释掉方法2上面的相关注解
 }

}

产生原因:

可能是由于静态方法是属于类的,而非静态方法是属于Bean的,该类会被加载到容器中。具体原因需要查资料,后续进行补充。

解决:

如work4,把你需要调用的静态方法放到非静态方法中进行调用。

补充:记录一次@Around使用不正确造成的StackOverflowError

同事发了一个底层服务后。我负责的某个上层服务突然挂了,有点慌,马上查看上层服务的日志。

查看日志特别长。没找到从哪儿报的。

跳到最上开始找,结果: StackOverflowError。就想到了递归调用。

随即找到相关代码一看:

由于doSomething方法也被@Around拦截,因此,一旦service下某个方法被拦截进入doAround,就会调用other.doSomething(),就又会触发doAround

导致一直递归调用且无法退出。

解决:

在@Around拦截的时候, 排除掉doSomething方法即可解决。

ps: @Around中不要用到被拦截方法。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。如有错误或未考虑完全的地方,望不吝赐教。

(0)

相关推荐

  • 如何动态修改JavaBean中注解的参数值

    我这里有一个需求需要修改Person类中的一个属性上的注解的值进行修改,例如: public class Person { private int age; @ApiParam(access="lala") private String name; //get set 方法忽略 } 将@ApiParam(access="lala") 修改为@ApiParam(access="fafa"),经过分析是可以实现的,需要用到动态代理进行操作. 具体源码

  • java 实现反射 json动态转实体类--fastjson

    我就废话不多说了,大家还是直接看代码吧~ package test.refect; public class Student { // 姓名 private String name; // 年龄 private String age; // 住址 private String address; public String getName() { return name; } public void setName(String name) { this.name = name; } public

  • Spring AOP 切面@Around注解的用法说明

    @Around注解可以用来在调用一个具体方法前和调用后来完成一些具体的任务. 比如我们想在执行controller中方法前打印出请求参数,并在方法执行结束后来打印出响应值,这个时候,我们就可以借助于@Around注解来实现: 再比如我们想在执行方法时动态修改参数值等 类似功能的注解还有@Before等等,用到了Spring AOP切面思想,Spring AOP常用于拦截器.事务.日志.权限验证等方面. 完整演示代码如下: 需要说明的是,在以下例子中,我们即可以只用@Around注解,并设置条件,

  • 创建动态代理对象bean,并动态注入到spring容器中的操作

    使用过Mybatis的同学,应该都知道,我们只需要编写mybatis对应的接口和mapper XML文件即可,并不需要手动编写mapper接口的实现.这里mybatis就用到了JDK动态代理,并且将生成的接口代理对象动态注入到Spring容器中. 这里涉及到几个问题.也许有同学会有疑问,我们直接编写好类,加入@Component等注解不是可以注入了吗?或者在配置类(@Configuration)中直接声明该Bean类型不也可以注入吗? 但具体到mybatis,这里我们用的是接口.由于spring

  • 解决@Around对静态方法不生效的问题

    场景: 在处理定时任务时,由于这几个方法都是静态方法,在aop的切面中使用@Around注解,进行监控方法调用是否有异常. 发现aop没有生效. 代码如下: /*切面类*/ @Aspect @Component public class RetryAop { private static Logger logger = LoggerFactory.getLogger(RetryAop.class); @Around(value = "@annotation(TechlogRetry)"

  • 解决php扩展安装不生效问题

    php安装扩展模块后,重启不生效的原因及解决办法 在lnmp运维环境中,我们经常会碰到有些php依赖的扩展模块没有安装,这就需要后续添加这些扩展模块.在扩展被安装配置后,往往会发现php-fpm服务重启后,这些扩展并没有真正加载进去!下面就以一个示例进行说明: 示例环境: php安装包存放路径:/data/software/php-5.6.10.tar.gz php安装目录:/data/php [root@zabbix ~]# ll /data/software/php-5.6.10.tar.g

  • 解决springboot引入swagger2不生效问题

    目录 问题描述: springboot引入swagger2的步骤: ①引入依赖 ②编写Swagger2的配置类 ③在controller中添加注解:按需添加注解 ④在model(pojo)上加注解,按需添加 一些注解的使用 今天遇到跟同事遇到一个由于失误导致的问题,也可以说比较难发现了.在此记录一下(我们用的springboot是2.0.3,swagger是2.2.2) 问题描述: swagger修改title,description等都不生效.并且启动springboot,没有有去加载swag

  • 解决Springboot配置excludePathPatterns不生效的问题

    目录 Springboot配置excludePathPatterns不生效 先说解决方案 使用excludePathPatterns过滤Swagger路径失败 Springboot配置excludePathPatterns不生效 先说解决方案 1.因为我在@RequestMapping中加了一个路径,所以  匹配原则应该是   /**/user/login/** 2.我的excludePathPatterns配置 3.分析:Interceptor加载关键所在!要在从url第一个"/"开

  • 解决shiro 定时监听器不生效的问题 onExpiration不调用问题

    问题 redis 抛出异常: redis.clients.jedis.ScanResult.getStringCursor()Ljava/lang/String; Method threw 'java.lang.NoSuchMethodError' exception. 说明 spring-boot 版本 <parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-

  • SpringBoot调用公共模块的自定义注解失效的解决

    目录 调用公共模块的自定义注解失效 项目结构如下 解决方法 SpringBoot注解不生效,踩坑 解决方法 调用公共模块的自定义注解失效 项目结构如下 我在 bi-common 公共模块里定义了一个自定义注解,实现AOP记录日志,bi-batch 项目已引用了 bi-common ,当在 bi-batch 使用注解的时候,没有报错,但是切面却失效. 自定义注解: @Target(ElementType.METHOD) @Retention(RetentionPolicy.RUNTIME) pub

  • java接口私有方法实现过程解析

    这篇文章主要介绍了java接口私有方法实现过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 问题描述: 我们需要抽取一个共有方法,用来解决两个默认方法之间重复代码的问题 但是这个共有方法不应该让实现类使用,应该是私有化的. 解决方案: 从java 9开始,接口当中允许定义私有方法. 1.普通私有方法,解决多个默认方法之间重复代码问题 格式: private 返回值类型方法名称(参数列表){ 方法体 } 2.静态私有方法,解决多个静态方法之

  • iOS手动添加新字体的步骤和踩坑记录

    目录 前言 字体添加 1.引入字体文件 2.配置 Info.plist 文件 3.解决添加新字体不生效问题 4.代码中设置字体 总结 前言 最近在一个日记软件,发现系统的默认字体过于丑陋,于是有了更改应用字体的想法.完成操作的过程中踩了一些小坑,写下此文记录一下,希望能对后人有所帮助. 字体添加 1.引入字体文件 直接将下载好的字体文件拖入项目中 2.配置 Info.plist 文件 在 Info.plist 文件中添加新行 "Fonts provided by application"

  • 详解Java接口的相关知识

    一.接口概述 接口,是Java语言中一种引用类型,是方法的集合,如果说类的内部封装了成员变量.构造方法.成员方法,那么接口的内部主要就是封装了方法,包含抽象方法(JDK 7及以前).默认方法和静态方法(JDK8),私有方法(JDK9). 二.定义格式 接口格式 public interface 接口名称 { // 抽象方法 // 默认方法 // 静态方法 // 私有方法 } 接口不能直接使用,必须有一个"实现类"来"实现接口". 实现类格式: public clas

  • 解决Cent0S 6.7直接在/etc/resolv.conf文件下修改DNS地址重启不生效问题

    CentOS 6.7/Linux下设置IP地址 1:临时修改: 1.1:修改IP地址 # ifconfig eth0 192.168.2.104 1.2:修改网关地址 # route add default gw 192.168.2.1 dev eth0 1.3:修改DNS # echo "nameserver 192.168.2.1" >> /etc/resolv.conf         // 与主机的DNS服务器保持一致 这个时候就可以上网了,上网的IP地址为192.

随机推荐