SpringBoot SpEL支持方法参数解析

想在IDEA中支持方法提示, 需要下载SpEL Assistant插件, 用法在这里

这里仅说AOP时支持方法参数

DemoAop.java:

java 复制代码
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface DemoAop {
    /**
     * 支持SpEL表达式
     */
    String value();
}

MyXXXAspect.java:

java 复制代码
@EnableAspectJAutoProxy
@Aspect
@Slf4j
public class CustomMetadataAspect implements BeanFactoryAware {

    private final ExpressionParser parser = new SpelExpressionParser();
    private final ParameterNameDiscoverer parameterNameDiscoverer = new DefaultParameterNameDiscoverer();
    @Setter
    private BeanFactory beanFactory;

    @Before("@annotation(demoAop)")
    public void atBefore(JoinPoint joinPoint, DemoAop demoAop) {
		String result = demoAop.value();
        MethodSignature signature = (MethodSignature) joinPoint.getSignature();
        Method targetMethod = signature.getMethod();
        Object[] args = joinPoint.getArgs();
        // noinspection DataFlowIssue
        MethodBasedEvaluationContext context =
            new MethodBasedEvaluationContext(null, targetMethod, args, pnd);
        context.setBeanResolver(new BeanFactoryResolver(beanFactory));
        // 可添加额外加参数, 如after切点添加result返回值: context.setVariable("额外参数", 参数值);
        result = parser.parseExpression(result).getValue(context, String.class);
        log.info("result is: {}", result);
    }

其实需要的代码行数特别少, 不需要花里胡哨的写法;

另外我一般会写上rootObject(MethodBasedEvaluationContext的第一个参数), 方便调用一些内容, 比如SpringSecurity中的@PreAuthorize中的hasRole()/hasAnyRole()/hasPermission()等等方法, 都是通过rootObject来实现的

示例我的一个RootObject:

java 复制代码
@Data
@AllArgsConstructor
@SuppressWarnings({"unused"})
public class RobinMetadataRootObject {
    private final Method method;

    private final Object[] args;

    private final Object target;

    private final Class<?> targetClass;

    public String getMethodName() {
        return this.method.getName();
    }
}
java 复制代码
// 解析时
public void atBefore(JoinPoint joinPoint, DemoAop demoAop) {
    ...
    // 初始化
    RobinMetadataRootObject rootObject = new RobinMetadataRootObject(targetMethod, extractArgs(targetMethod, args),
            joinPoint.getTarget(), joinPoint.getTarget().getClass());
    // 赋值
    MethodBasedEvaluationContext context =
            new MethodBasedEvaluationContext(rootObject, targetMethod, args, parameterNameDiscoverer);
    ...
}

可以方便的取方法和函数名称

Spring中的一些其他rootObject, 可供参考

Cache: CacheExpressionRootObject

Security: SecurityExpressionRoot

一些常见SpEL用法, 这里仅做示例, rootObject以, 不做详细解释:

java 复制代码
public String test(String id, Integer[] idList) {
    return id;
}
  • "#id" -- 直接取值
  • "1 > 2 && 3 == 4 && 5 > 6"关系运算符, 支持> / < / >= / <= / !=, 其中null小于任何值
  • 运算符支持纯字母等效替代: lt ('<')、gt ('>')、le ('<=')、ge ('>=')、eq ('==')、ne ('!= ')、div ('/')、mod ('%')、not ('!') 不区分大小写
  • "#id.startsWith('a')" -- 调用一些方法
  • "#id?.startsWith('b')" -- 空安全
  • "getMethodName() + getMethod() + getArgs()[0] + #args[0]" -- root上的字段/方法
  • "#p0+#a0" -- 以pa前缀 + 参数序号来取值
  • "@taskExecutor.activeCount" -- 使用@符号 + bean名称调用其他bean
  • "1 + 2 - 3 * 4 / 5 - -6 % 7" -- 加减乘除, 正负数, 取余
  • "'abc' + true + null + 123" -- 各种字面量
  • "{1, 2, 3, 4, 5}.?[ #this > 3 ]" -- 内联列表以及过滤(返回值为[4,5])
  • "{1, 2, 3, 4, 5}.^[ #this > 1 ]" -- 查找第一个符合条件的值(结果为2)
  • "{1, 2, 3, 4, 5}.&[ #this > 1 ]" -- 查找最后一个符合条件的值(结果为5)
  • "#username ?: 'abcd'" -- 默认值 如果username为 null, 则赋值为"abcd"
  • "new String(234)" -- 创建对象
  • "true ? 1 : 2" -- 三元表达式
相关推荐
客卿12313 分钟前
滑动窗口--模板
java·算法
zjjsctcdl36 分钟前
java与mysql连接 使用mysql-connector-java连接msql
java·开发语言·mysql
Moe4881 小时前
WebSocket :从浏览器 API 到 Spring 握手、Handler 与前端客户端
java·后端·架构
顶点多余1 小时前
线程互斥+线程同步+生产消费模型
java·linux·开发语言·c++
⑩-1 小时前
Java基础+集合框架-八股文
java·开发语言
福运常在1 小时前
股票数据API(19)次新股池数据
java·python·maven
Zaki_gd1 小时前
Cortex-M7 D-Cache 与 DMA 缓存一致性说明
java·spring·缓存
多看书少吃饭1 小时前
Vue3 + Java + Python 打造企业级大模型知识库(含 SSE 流式对话完整源码)
java·python·状态模式
Arthas2171 小时前
Java大厂面试:从Spring到微服务的全面技术考察
java·jvm·spring·微服务·面试·并发
mifengxing1 小时前
力扣HOT100——(1)两数之和
java·数据结构·算法·leetcode·hot100