栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 软件开发 > 后端开发 > Java > SpringBoot

如何通过aop+spel表达式玩转出不一样的切面实现

SpringBoot 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

如何通过aop+spel表达式玩转出不一样的切面实现

前言

在介绍正文前,我们先来讲下spel

什么是spel

Spring表达式语言(简称“ SpEL”)是一种功能强大的表达式语言,支持在运行时查询和操作对象图。

语言语法类似于Unified EL,但是提供了其他功能,最著名的是方法调用和基本的字符串模板功能。

此外它并不直接与Spring绑定,而是可以独立使用

spel可以支持哪些功能
  • 文字表达式
  • 布尔运算符和关系运算符
  • 常用表达式
  • 类表达式
  • 访问属性,数组,列表和映射
  • 方法调用
  • 关系运算符
  • 分配
  • 调用构造函数
  • Bean引用
  • 数组构造
  • 内联列表
  • 内联Map
  • 三元运算符
  • 变量
  • 用户定义的功能
  • 集合投影
  • 集合选择
  • 模板表达式

上述的spel语法可以通过如下链接进行查阅
docs.spring.io/spring-framework/docs/current/reference/html/core.html#expressions-language-ref

spel解析基本流程

形如下图

大体的步骤如下

  1. 创建解析器
  2. 解析表达式
  3. 构造上下文
  4. 求值
spel核心接口介绍

1、org.springframework.expression.expressionParser

表达式解析器,其功能主要是将字符串表达式转换为expression对象。支持解析模板以及标准表达式字符串

其默认实现为

org.springframework.expression.spel.standard.SpelexpressionParser

2、org.springframework.expression.evaluationContext

spel计算表达式值的“上下文”,这个Context对象可以包含多个对象,但只能有一个root(根)对象。当表达式中包含变量时,spel会根据evaluationContext中的变量的值对表达式进行计算。可以使用setRootObject方法来设置根对象,使用setVariable方法来注册自定义变量,使用registerFunction来注册自定义函数。

其默认实现为

org.springframework.expression.spel.support.StandardevaluationContext

3、org.springframework.expression.expression

代表一个表达式,通过getValue方法根据上下文获得表达式值

其默认实现为

org.springframework.expression.spel.standard.Spelexpression
spel官方文档

docs.spring.io/spring-framework/docs/current/reference/html/core.html#expressions

正文

前边简要介绍一下spel,下边我们就通过一个小例子来演示下。

该小例子主要是通过AOP+SPEL来实现,例子场景是:当产品价格大于10时,放入本地缓存,并通过定时器打印出本地缓存的值

1、业务逻辑实现核心代码

@Service
public class ProductServiceImpl implements ProductService {



    @Autowired
    private ProductMockDao productMockDao;

    @Override
    @LocalCacheable(key = "#product.id",condition = "#product.price ge 10")
    public Product save(Product product) {
 return productMockDao.save(product);
    }


}

2、aop切面编写

@Component
@Aspect
public class CacheAspect {


    @Around("@annotation(localCacheable)")
    public Object around(ProceedingJoinPoint pjp, LocalCacheable localCacheable) throws Throwable{
 MethodSignature methodSignature = (MethodSignature)pjp.getSignature();
 Method method = methodSignature.getMethod();
 Object[] args = pjp.getArgs();
 Object result = pjp.proceed();
 String key = pjp.getTarget().getClass().getName() + "_" + method.getName() + "_" + args.length;

 if(!StringUtils.isEmpty(localCacheable.key())){
    key = SpELParserUtils.parse(method,args,localCacheable.key(),String.class);
 }

 System.out.println("key:"+key);

 if(!StringUtils.isEmpty(localCacheable.condition())){
     boolean condition = SpELParserUtils.parse(method,args,localCacheable.condition(),Boolean.class);
     if(condition){
  LocalCache.INSTANCE.put(key,result);
     }
 }else{
     LocalCache.INSTANCE.put(key,result);
 }

 return result;

    }
}

3、解析spel核心工具类

@Slf4j
public final class SpELParserUtils {

    private static final String expression_PREFIX = "#{";

    private static final String expression_SUFFIX = "}";

    
    private static expressionParser expressionParser = new SpelexpressionParser();

    
    private static DefaultParameterNameDiscoverer parameterNameDiscoverer = new DefaultParameterNameDiscoverer();



    private SpELParserUtils(){}

    
    public static  T parse(Method method, Object[] args, String spelexpression, Class clz, T defaultResult) {
 String[] params = parameterNameDiscoverer.getParameterNames(method);
 evaluationContext context = new StandardevaluationContext();
 //设置上下文变量
 for (int i = 0; i < params.length; i++) {
     context.setVariable(params[i], args[i]);
 }
 T result = getResult(context,spelexpression,clz);
 if(Objects.isNull(result)){
     return defaultResult;
 }
 return result;
    }

    
    public static  T parse(Method method, Object[] args, String spelexpression, Class clz) {
 String[] params = parameterNameDiscoverer.getParameterNames(method);
 evaluationContext context = new StandardevaluationContext();
 //设置上下文变量
 for (int i = 0; i < params.length; i++) {
     context.setVariable(params[i], args[i]);
 }
 return getResult(context,spelexpression,clz);
    }

    
    public static  T parse(String param, Object paramValue, String spelexpression, Class clz) {
 evaluationContext context = new StandardevaluationContext();
 //设置上下文变量
 context.setVariable(param, paramValue);
 return getResult(context,spelexpression,clz);
    }


    
    public static  T parse(String param, Object paramValue,String spelexpression, Class clz, T defaultResult) {
 evaluationContext context = new StandardevaluationContext();
 //设置上下文变量
 context.setVariable(param, paramValue);
 T result = getResult(context,spelexpression,clz);
 if(Objects.isNull(result)){
     return defaultResult;
 }
 return result;

    }


    
    private static  T getResult(evaluationContext context,String spelexpression, Class clz){
 try {
     //解析表达式
     expression expression = parseexpression(spelexpression);
     //获取表达式的值
     return expression.getValue(context, clz);
 } catch (Exception e) {
     log.error(e.getMessage(),e);
 }
 return null;
    }


    
    private static expression parseexpression(String spelexpression){
 // 如果表达式是一个#{}表达式,需要为解析传入模板解析器上下文
 if(spelexpression.startsWith(expression_PREFIX) && spelexpression.endsWith(expression_SUFFIX)){
     return expressionParser.parseexpression(spelexpression,new TemplateParserContext());
 }

 return expressionParser.parseexpression(spelexpression);
    }

}

4、 示例效果

总结

spel在spring应用中随处可见,比如@cacheable、@Value等,我们也可以通过aop+spel实现出适合我们业务场景的功能

demo链接

github.com/lyb-geek/springboot-learning/tree/master/springboot-aop-spel

转载请注明:文章转载自 www.mshxw.com
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号