自定义注解动态获取接口参数

自定义注解动态获取接口参数

需求:

用户访问企业详情时候,校验用户剩余查看次数,剩余次数为0则不能访问详情接口。

实现方案:

自定义注解+切面实现对用户权限的校验。

假设存在如下访问企业详情的接口:

@GetMapping("/getDetail")
// 配合SpEL 表达式  "#id"
@AuthUserPermission(key="#id")
public ResponseData getCompanyDetail(@RequestParam("id") Long id){
// todo someThing
}

自定义注解:

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface AuthUserPermission {
   // 用来接受接口上的动态参数
    String key();
}

自定义切面:

@Component
@Slf4j
@Aspect
public class AuthUserPermissionAspect {
	// 切入点
	@Pointcut("@annotation(com.boshiyun.application.anotation.AuthUserPermission)")
	public void pointcut(){}

    @Around("pointcut()")
    public Object around(ProceedingJoinPoint point)  {
    // 校验用户权限
        beforeAround(point);
        try {
            Object proceed = point.proceed();
            return proceed;
        } catch (Throwable e) {
            throw new RunTimeException(e);
        }
    }

    public void beforeAround(JoinPoint joinPoint){
     // 在这获取接口的参数 
      String id= getId(joinPoint);
      if(StringUtils.isBlank(id)){
      	throw new RunTimeException("参数错误")
      }
      // 并校验用户权限,如果用户无权限则抛出异常
    }
        public String getId(JoinPoint joinPoint){
        MethodSignature signature = (MethodSignature)joinPoint.getSignature();
        Method method = signature.getMethod();
        AuthUserPermission annotation = method.getAnnotation(AuthUserPermission.class);
        if(annotation ==null){
            return "";
        }
        if(StringUtils.isBlank(annotation.key())){
            return "";
        }
        String strValue = generateKeyBySpEL(annotation.key(), joinPoint);
        if(StringUtils.isBlank(strValue)){
            return "";
        }
           return strValue ;
    }

// 通过SpEL表达式获取到传入的 id
    private SpelExpressionParser parserSpel = new SpelExpressionParser();
    private DefaultParameterNameDiscoverer parameterNameDiscoverer= new DefaultParameterNameDiscoverer();
   
    public String generateKeyBySpEL(String key, JoinPoint pjp) {
        Expression expression = parserSpel.parseExpression(key);
        EvaluationContext context = new StandardEvaluationContext();
        MethodSignature methodSignature = (MethodSignature) pjp.getSignature();
        Object[] args = pjp.getArgs();
        String[] paramNames = parameterNameDiscoverer.getParameterNames(methodSignature.getMethod());
        for(int i = 0 ; i < args.length ; i++) {
            context.setVariable(paramNames[i], args[i]);
        }
        return expression.getValue(context).toString();
    }
}

以上获取参数方法亲测可行,如果是其他接口传入的是对象,则接口上注解修改为

@AuthUserPermission(key="#c.id")
public ResponseData getCompanyDetail(Compnay c){

}

当然也可以使用在其切面中直接通过JoinPoint直接 获取到参数,不过应对诸多接口 有些地方 企业id的字段名称为 id 或者 companyId,这样如果使用 JoinPoint 获取参数,我们不知道某个接口企业id对应的是 id,或者 companyId, 所以 通过注解和 sepl 表达式,只需要在接口注解上指定我需要获取的是哪个字段名称对应的值就可以了

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
自定义注解可以用来获取请求参数。可以通过实现HandlerMethodArgumentResolver接口来自定义解析器,该解析器用于处理特定的参数注解。在该解析器中,可以使用requestbody来接收请求参数,并通过二次取值来获取对应的值。例如,可以使用JsonField注解来指定需要获取参数值。在解析器中,可以通过读取请求体的内容,并使用JSONPath来解析出对应的值。 以下是一个示例代码,演示了如何使用自定义注解获取请求参数: ```java public class JsonFieldArgumentResolver implements HandlerMethodArgumentResolver { @Override public boolean supportsParameter(MethodParameter parameter) { return parameter.hasParameterAnnotation(JsonField.class); } @Override public Object resolveArgument(MethodParameter parameter, ModelAndViewContainer mavContainer, NativeWebRequest webRequest, WebDataBinderFactory binderFactory) throws Exception { HttpServletRequest request = (HttpServletRequest) (webRequest.getNativeRequest()); BufferedReader br = request.getReader(); String str, reqStr = ""; while ((str = br.readLine()) != null) { reqStr = str; } return JSONPath.read(reqStr, "$." + parameter.getParameterAnnotation(JsonField.class).value()); } } ``` 在Controller中,可以使用@JsonField注解来指定需要获取参数值。例如: ```java @RequestMapping(value = "/test", method = RequestMethod.POST) public ResponseResult test(@JsonField("name") String name) { System.out.println("request: " + name); return new ResponseResult(ResponseCode.SUCCESS.getCode(), ResponseCode.SUCCESS.getMsg(), name); } ``` 这样,在请求中传递的参数会被解析器自动处理,并将对应的值注入到方法参数中。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [自定义注解实现获取requestbody中指定参数](https://blog.csdn.net/m0_46444563/article/details/124235335)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值