springboot: spel结合aop实现动态传参

本文介绍如何利用Spring Expression Language (SpEL) 和面向切面编程 (AOP) 实现动态参数传递,包括SpEL的基本概念、核心代码实现及一个具体的使用案例。

前言

基于SpingBoot框架中, 我们随处可以见的便是各种各样的功能注解, 注解的实现原理AOP之前有说过(翻看本系列的前面几章即可), 这里不过多赘述.

那么, 你有没有碰到这样一种场景: 需要动态的传参数进注解, 注意是动态的而不是写死在代码里的.

针对这种需求, 今天, 我们就来实现一个简单的案例.

SpEl表达式简介

正式撸代码之前, 先了解下SpEl (Spring Expression Language) 表达式, 这是Spring框架中的一个利器.

Spring通过SpEl能在运行时构建复杂表达式、存取对象属性、对象方法调用等等.

举个简单的例子方便理解, 如下

//定义了一个表达式
String expressionStr = "1+1";
ExpressionParser parser = new SpelExpressionParser();
Expression expression = parser.parseExpression(expressionStr);
Integer val = expression.getValue(Integer.class);
System.out.println(expressionStr + "的结果是:" + val);

通过以上案例, 不难理解, 所谓的SpEl, 本质上其实就是解析表达式,.

关于SpEl表达式感兴趣的可以自行查阅资料, 本篇不做细致的讨论.

实例: SpEl结合AOP动态传参

简单了解了SpEl表达式, 那么接下来我们就直接开始撸代码.

先引入必要的pom依赖, 其实只有aop依赖, SpEl本身就被Spring支持, 所以无需额外引入.

<dependency>
          <groupId>org.springframework.boot</groupId>
          <artifactId>spring-boot-starter-aop</artifactId>
</dependency>

定义一个SpEl的工具类SpelUtil

publicclass SpelUtil {

    /**
     * 用于SpEL表达式解析.
     */
    private static final SpelExpressionParser parser = new SpelExpressionParser();

    /**
     * 用于获取方法参数定义名字.
     */
    private static final DefaultParameterNameDiscoverer nameDiscoverer = new DefaultParameterNameDiscoverer();

    /**
     * 解析SpEL表达式
     *
     * @param spELStr
     * @param joinPoint
     * @return
     */
    public static String generateKeyBySpEL(String spELStr, ProceedingJoinPoint joinPoint) {
        // 通过joinPoint获取被注解方法
        MethodSignature methodSignature = (MethodSignature) joinPoint.getSignature();
        Method method = methodSignature.getMethod();
        // 使用Spring的DefaultParameterNameDiscoverer获取方法形参名数组
        String[] paramNames = nameDiscoverer.getParameterNames(method);
        // 解析过后的Spring表达式对象
        Expression expression = parser.parseExpression(spELStr);
        // Spring的表达式上下文对象
        EvaluationContext context = new StandardEvaluationContext();
        // 通过joinPoint获取被注解方法的形参
        Object[] args = joinPoint.getArgs();
        // 给上下文赋值
        for (int i = 0; i < args.length; i++) {
            context.setVariable(paramNames[i], args[i]);
        }
        // 表达式从上下文中计算出实际参数值
        /*如:
            @annotation(key="#user.name")
            method(User user)
             那么就可以解析出方法形参的某属性值,return “xiaoming”;
          */
        return expression.getValue(context).toString();
    }
}

定义一个带参注解SpelGetParm

@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
public @interface SpelGetParm {

    String parm() default "";

}

定义带参注解SpelGetParmAop

@Aspect
@Slf4j
@Component
public class SpelGetParmAop {

    @PostConstruct
    public void init() {
        log.info("SpelGetParm init ......");
    }
    /**
     * 拦截加了SpelGetParm注解的方法请求
     *
     * @param joinPoint
     * @param spelGetParm
     * @return
     * @throws Throwable
     */
    @Around("@annotation(spelGetParm)")
    public Object beforeInvoce(ProceedingJoinPoint joinPoint, SpelGetParm spelGetParm) throws Throwable {
        Object result = null;
        // 方法名
        String methodName = joinPoint.getSignature().getName();
        //获取动态参数
        String parm = SpelUtil.generateKeyBySpEL(spelGetParm.parm(), joinPoint);
        log.info("spel获取动态aop参数: {}", parm);
        try {
            log.info("执行目标方法: {} ==>>开始......", methodName);
            result = joinPoint.proceed();
            log.info("执行目标方法: {} ==>>结束......", methodName);
            // 返回通知
            log.info("目标方法 " + methodName + " 执行结果 " + result);
        } finally {

        }
        // 后置通知
        log.info("目标方法 " + methodName + " 结束");
        return result;
    }

以上已经基本实现了案例的核心功能, 接下来我们使用该注解即可

定义一个实体User

@Getter
@Setter
@NoArgsConstructor
@JsonSerialize
@JsonInclude(Include.NON_NULL)
public class User implements Serializable {
    private static final long serialVersionUID = -1941635009126453180L;

    private String name;
    private Long id;

}

我们在UserController直接使用该带参注解即可

@CrossOrigin
@RestController
@RequestMapping("/user")
public class UserController {
    @PostMapping("/param")
    @SpelGetParm(parm = "#user.name")
    public R repeat(@RequestBody User user) {
        return R.success(user);
    }
}

最后请求

image
image

可以看出, 切面成功获取到了实体的name值“张三”.

小结

结合SpEl表达式可以实现各种“骚操作”, 各位大佬可自由发挥, 下面一章我们准备结合SpEl来实现分布式锁的功能.

<think> 首先,用户的问题是:“我想查找在Spring Boot中使用SpEL动态实现大小比较的方法”。用户提供了三个引用,我需要参考这些引用来回答。引用[1]讨论了SpEL用于日志记录,引用[2]是关于SpEL的快速入门,引用[3]是关于使用SpEL动态生成邮件内容,包括数据对比。 用户指定:“上述引用仅供参考,并不是用户真正的需求,请尽可能多的参考用户上一次的问题和你的回答。” 但在这个对话中,没有显示上一次的问题和回答。我应该基于当前的问题和提供的引用来回答。 关键点是:用户想知道如何在Spring Boot中使用SpEL动态实现大小比较。大小比较可能涉及比较数值或对象的大小。 从引用中提取相关信息: - 引用[1]提到SpEL使用`#{...}`作为定界符,可以引用bean的属性、调用方法等。 - 引用[2]介绍SpEL是一种表达式语言,支持运行时查询和操作对象图,包括方法调用。 - 引用[3]描述使用SpEL进行数据横向对比和异常高亮,这暗示了比较操作。 SpEL支持比较运算符,如`>`、`<`、`>=`、`<=`、`==`等。用户可以在表达式中直接使用这些运算符来进行大小比较。 在Spring Boot中,使用SpEL的常见方式包括: - 在注解中使用,如`@Value`注解。 - 在Spring Security表达式中。 - 在模板或配置中动态评估表达式。 回答结构: 1. **介绍SpEL的比较功能**:解释SpEL支持比较运算符。 2. **提供示例代码**:展示如何在Spring Boot中使用SpEL进行大小比较。 3. **参考引用**:自然地引用提供的材料。 4. **生成相关问题**:按照系统指令,在末尾添加相关问题。 系统指令要求: - 行内数学表达式用`$...$`格式(但这里可能不涉及数学表达式)。 - 独立公式用`$$...$$`(不适用)。 - LaTeX语法正确(确保代码块正确)。 - 使用中文回答。 - 生成相关问题。 - 引用段落末尾添加引用标识。 - 回答结构清晰。 回答大纲: - 开头:确认问题并简要解释。 - 主体:详细描述方法,包括示例代码。 - 结尾:总结并添加相关问题。 示例代码:使用Spring Boot的一个简单示例,比如在控制器或服务中使用SpEL进行大小比较。 可能的代码示例: - 使用`SpelExpressionParser`来解析和评估表达式。 - 在`@Value`注解中使用SpEL进行大小比较。 例如: ```java import org.springframework.expression.Expression; import org.springframework.expression.ExpressionParser; import org.springframework.expression.spel.standard.SpelExpressionParser; public class SpelDemo { public static void main(String[] args) { ExpressionParser parser = new SpelExpressionParser(); Expression exp = parser.parseExpression("10 > 5"); Boolean result = (Boolean) exp.getValue(); System.out.println(result); // 输出 true } } ``` 在Spring Boot中,可以在bean中使用: ```java @Value("#{10 > 5}") private boolean isGreater; ``` 或者动态比较变量: ```java @Value("#{someBean.value > 10}") private boolean isValueGreater; ``` 参考引用:引用[3]提到使用SpEL进行数据对比,这涉及到比较。可以说:“如引用[3]中所述,SpEL可以用于数据横向对比[^3],这包括大小比较。” 相关问题生成:基于主题,生成相关的问题。 例如: 1. SpEL支持哪些比较运算符? 2. 如何在Spring Boot动态传递参数到SpEL表达式进行比较? 3. SpEL在异常处理中的应用? 确保语言中文。 </think> ### 在Spring Boot中使用SpEL动态实现大小比较的方法 在Spring Boot中,使用SpELSpring Expression Language)动态实现大小比较是一种高效的方式。SpEL支持标准的比较运算符(如`>`、`<`、`>=`、`<=`、`==`),允许在运行时评估表达式,并根据条件返回布尔值。这常用于动态配置、条件
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值