有一段代码 有时会出现类型转换异常 很诡异 排查原因发现是spring aop造成的。
项目中我使用了aop进行自定义权限,若权限不通过 返回固定的ResponseVo 对应字段为:
@Data
@AllArgsConstructor
@NoArgsConstructor
public class ResponseVo<T> {
@JsonProperty("response_code")
private Integer responseCode;
@JsonProperty("verbose_msg")
private String verboseMsg;
private T data;
}
为了开发方便 不需要每个controller都写一个ResponseVo的构建 使用了ResponseBodyAdvice来帮助构建,之前的文章里写过
说白了就说Controller方法只需要返回一个任意对象 由spring将其封装到ResponseVo的data字段中。
下面的是使用ResponseBodyAdvice的效果 上面是正常写法
这个接口比较简单可能没太大对比度
但是在鉴权的aop中 若权限通过则调用方法 返回方法结果
若不通过 则返回内容为: 是一个ResponseVo
报错内容为 不能把ResponseVo转为BaseInfoVo,且堆栈中看不出任何自己写的代码引用。
测试时发现 有权限时 这个接口不报错,没有权限时就会报类型转换异常。找到这个规律 就很容易知道问题出在哪了。
原因是: spring在代理这个方法时 知道这个方法声明的返回值为BaseInfoVo,但当权限不通过时 我通过切面将其返回值改为了ResponseVo,但R不能转为B 就会报类型转换异常。
解决办法有2种:
1 Controller的返回类型统一为ResponseVo,这样就没办法使用ResponseBodyAdvice了。
2 将方法的返回值声明为所有可能被返回类型的共同接口:1 Object 因为Object是所有类的父类 2或声明一个接口由R和Q分别实现 但这样做意义不大。 缺点是返回值都为Object的话 语义不明确
最终选用了方法1