自定义注解支持SpEL表达式(动态参数)

实际生产中一个案例:利用AOP生成用户操作日志

代码直接复制过去就可以测试,亲测可用

1.定义日志注解

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface SysLog {

    //普通的操作说明
    String value() default "";
    
    //spel表达式的操作说明
    String spelValue() default "";
}

2.定义spel解析工具类

public class SpelUtil {

    /**
     * 用于SpEL表达式解析.
     */
    private static SpelExpressionParser parser = new SpelExpressionParser();
    /**
     * 用于获取方法参数定义名字.
     */
    private static DefaultParameterNameDiscoverer nameDiscoverer = new DefaultParameterNameDiscoverer();

    public static String generateKeyBySpEL(String spELString, ProceedingJoinPoint joinPoint) {
        // 通过joinPoint获取被注解方法
        MethodSignature methodSignature = (MethodSignature) joinPoint.getSignature();
        Method method = methodSignature.getMethod();
        // 使用spring的DefaultParameterNameDiscoverer获取方法形参名数组
        String[] paramNames = nameDiscoverer.getParameterNames(method);
        // 解析过后的Spring表达式对象
        Expression expression = parser.parseExpression(spELString);
        // 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="#student.name")
             method(Student student)
             那么就可以解析出方法形参的某属性值,return “xiaoming”;
          */
        return expression.getValue(context).toString();
    }
}

3.定义切面类

@Aspect
@Component
public class SysLogAspect {
    @Autowired
    private LogService logService;

    @Autowired
    private HttpServletRequest request;

    @Pointcut("@annotation(com.ztri.common.annotation.SysLog)")
    public void logPointCut() {

    }

    @Around("logPointCut()")
    public Object around(ProceedingJoinPoint point) throws Throwable {
        long beginTime = System.currentTimeMillis();
        //执行方法
        Object result = point.proceed();
        //执行时长(毫秒)
        long time = System.currentTimeMillis() - beginTime;

        //保存日志
        saveSysLog(point, time);

        return result;
    }
    private void saveSysLog(ProceedingJoinPoint joinPoint, long time) {
        MethodSignature signature = (MethodSignature) joinPoint.getSignature();
        Method method = signature.getMethod();

        Log sysLog = new Log();
        sysLog.setTime(time);
        SysLog syslog = method.getAnnotation(SysLog.class);
        if (syslog != null) {
            //注解上的描述
            if (StrUtil.isNotBlank(syslog.value())) {
                sysLog.setOperation(syslog.value());
            }
            if (StrUtil.isNotBlank(syslog.spelValue())) {
                String spelValue = SpelUtil.generateKeyBySpEL(syslog.spelValue(), joinPoint);
                sysLog.setOperation(spelValue);
            }
        }
        //请求的方法名
        String className = joinPoint.getTarget().getClass().getName();
        String methodName = signature.getName();
        sysLog.setMethod(className + "." + methodName + "()");

        //请求的参数
        Object[] args = joinPoint.getArgs();
        try {
            String params = JSONUtil.toJsonStr(args);
            sysLog.setParams(params);
        } catch (Exception e) {

        }

        //设置IP地址
        sysLog.setIp(ServletUtil.getClientIP(request));
        UserAgent ua = UserAgentUtil.parse(request.getHeader("User-Agent"));
        sysLog.setBrowser(ua.getBrowser().toString());

        //保存系统日志
        logService.create(sysLog);
    }
}

4.方法上使用日志注解

    @ApiOperation("高级搜索(包含点击1.热门列表 2.更多跳转页面)")
    @PostMapping("searchData")
    @SysLog(spelValue = "'高级搜索' + #searchVo.keyWord")
    public ResponseEntity<Object> searchData(@RequestBody SearchVo searchVo) throws IOException {
        SearchDto searchDto = searchService.searchData(searchVo);
        return new ResponseEntity<>(searchDto, HttpStatus.OK);
    }

    @ApiOperation("登录授权")
    @PostMapping("/login")
    @SysLog("用户登录")
    public ResponseEntity<Object> login(@Validated(User.Create.class) @RequestBody LoginUser loginUser) {

        return ResponseEntity.ok(authInfo);
    }

  • 1
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
好的,你的问题是如何动态命名@Service注解的value值。在Spring,我们可以使用SpEL表达式来动态地设置@Service注解的value值。 SpELSpring表达式语言,它提供了一种在运行时使用表达式的方式,可以在运行时计算出表达式的值。我们可以使用SpEL表达式来设置@Service注解的value值,例如: ``` @Service("#{myServiceConfig.serviceName}") public class MyService { // ... } ``` 在这个例子,我们使用了一个SpEL表达式来动态地设置@Service注解的value值。表达式`#{myServiceConfig.serviceName}`的myServiceConfig是一个Spring配置文件定义的bean,它包含一个名为serviceName的属性,我们将这个属性的值用作@Service注解的value值。 我们可以在Spring配置文件定义这个bean,例如: ``` <bean id="myServiceConfig" class="com.example.MyServiceConfig"> <property name="serviceName" value="myService"/> </bean> ``` 在这个例子,我们定义了一个名为myServiceConfig的bean,它是一个自定义类MyServiceConfig的实例,包含一个名为serviceName的属性,其值为"myService"。这个bean可以在我们的服务类使用SpEL表达式来动态地设置@Service注解的value值。 需要注意的是,如果我们使用SpEL表达式来动态命名@Service注解的value值,我们需要在Spring配置文件启用SpEL解析器。我们可以通过在配置文件添加以下配置来实现: ``` <bean class="org.springframework.context.expression.PropertyPlaceHolderConfigurer"/> ``` 这样,我们就可以动态地命名@Service注解的value值,并在应用程序注入我们的服务类。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

zhangkaixuan456

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值