前言
为了能在SpringBoot使用Kotlin,真的是各种坑都彩礼一遍,这次遇到的问题是Aspect无法对Kotlin代码生效。我这里的使用场景是使用切面切Controller中的方法,用来对接口进行一些初始化和收尾工作。
Aspect在Controller类还是Java代码的时候是生效的,能够正常切入,但是通过IDEA自带的“Convert Java File to Kotlin File”后,转为Kotlin类的Controller就无法被切面切入了,搞得我一头雾水。
Controller中的代码如下:
解决过程
我首先想到的是将切的controller类改为切注解,也就是Swagger的@ApiOperation注解,因为我的每个Controller中的方法涉及到接口的都有这个注解,结果自然无济于事。
然后我怀疑是否Aspect类也必须是Kotlin写的,于是把我的Aspect类也转为了Kotlin,结果发现还是不行。Aspect代码如下:
@Pointcut("execution(public * com.itdct.server..*.controller..*(..))")
fun pointCut() {
}
@Around("pointCut()")
@Throws(Throwable::class)
fun around(joinPoint: ProceedingJoinPoint): Any {
val args = joinPoint.args
val arg = args[0] ?: throw CommonException(ReturnCode.LACK_PARAM, "请求对象不存在")
// INFO: DCT: 2023/9/2 初始化整个接口的上下文
val context = initContext(arg, joinPoint)
// INFO: DCT: 2023/9/2 检查Token信息
checkToken(context, arg)
// INFO: DCT: 2023/9/2 根据请求的类型初始化不同操作,并对Cache初始化
initByUrlType(context)
// INFO: DCT: 2023/9/3 执行controller中的方法,如果有Cache会直接返回
val resultObj = processArgs(context, joinPoint)
// INFO: DCT: 2023/9/3 处理返回参数,以及处理Cache
handleResponse(context, resultObj)
// INFO: DCT: 2023/9/3 当全部完成
onFinish(context)
return resultObj
}
然后我把@Pointcut中的public都去掉了,这下出现了一个很奇怪的事,controller中的方法没切到,但是把BaseController中用Java写的protected修饰的getContext方法给切到了。
protected Context getContext() {
Context context = threadContextMap.get(Thread.currentThread());
return context;
}
这就说明Aspect对Kotlin肯定是生效的!肯定是什么地方出了点问题,然后我看了一下Kotlin编译后的java字节码,答案不用猜都知道了,方法被final给修饰了!!
这也说明了Aspect无法对final修饰的方法生效,而Kotlin中,不加open的方法默认被final修饰,这点我没反应过来,导致这个问题坑了我很久。
解决方法
解决方法很简单,在controller中的方法加一个open就行了