【错误记录】Groovy 函数拦截调用 invokeMethod 导致栈溢出 ( java.lang.StackOverflowError )

2023-03-30 10:38:00 浏览数 (1)

文章目录

  • 一、报错信息
  • 二、解决方案

一、报错信息


使用 Groovy 函数拦截功能 , 定义 Groovy 类 , 实现 GroovyInterceptable 接口 , 并重写 invokeMethod 方法 , 在该方法中使用

代码语言:javascript复制
println "invokeMethod"

代码 , 打印日志 ;

完整代码如下 :

代码语言:javascript复制
class Student implements GroovyInterceptable{
    def name;

    def hello() {
        println "Hello ${name}"
    }

    @Override
    Object invokeMethod(String name, Object args) {
        println "invokeMethod"
        //System.out.println "invokeMethod"
    }
}

def student = new Student(name: "Tom")

// 直接调用 hello 方法
student.hello()

报错信息 :

代码语言:javascript复制
Caught: java.lang.StackOverflowError
java.lang.StackOverflowError
	at Student.invokeMethod(Groovy.groovy:10)
	at Student.invokeMethod(Groovy.groovy:10)
	at Student.invokeMethod(Groovy.groovy:10)
	at Student.invokeMethod(Groovy.groovy:10)
	at Student.invokeMethod(Groovy.groovy:10)
	at Student.invokeMethod(Groovy.groovy:10)
	at Student.invokeMethod(Groovy.groovy:10)
	at Student.invokeMethod(Groovy.groovy:10)
	at Student.invokeMethod(Groovy.groovy:10)
	at Student.invokeMethod(Groovy.groovy:10)
	at Student.invokeMethod(Groovy.groovy:10)
	at Student.invokeMethod(Groovy.groovy:10)
	at Student.invokeMethod(Groovy.groovy:10)

二、解决方案


调用 实现了 GroovyInterceptable 接口的 Student 类的 hello 方法 , 会调用 invokeMethod 方法 ,

在 invokeMethod 方法中 , 又调用了 println 方法 ,

代码语言:javascript复制
    @Override
    Object invokeMethod(String name, Object args) {
        println "invokeMethod"
        //System.out.println "invokeMethod"
    }

println 方法是 Groovy 注入到 Object 对象中的 , 在 Student 对象中 , 调用 println 也会回调 invokeMethod 方法 , 而在 invokeMethod 方法中又调用了 println 方法 , 这样循环调用 , 最终导致栈溢出 ;

在 invokeMethod 中 , 不调用 println 方法 , 调用 System.out.println 进行日志打印 , 这样就可以避免栈溢出 ;

代码语言:javascript复制
class Student implements GroovyInterceptable{
    def name;

    def hello() {
        println "Hello ${name}"
    }

    @Override
    Object invokeMethod(String name, Object args) {
        System.out.println "invokeMethod"
    }
}

def student = new Student(name: "Tom")

// 直接调用 hello 方法
student.hello()

0 人点赞