springBoot之自定义注解

自定义注解开发:

使用@interface关键字, 其定义过程与定义接口非常类似, 需要注意的是:
Annotation的成员变量在Annotation定义中是以无参的方法形式来声明的, 其方法名和返回值类型定义了该成员变量的名字和类型, 而且我们还可以使用default关键字为这个成员变量设定默认值

枚举:

常量的集合,枚举类的实例全部在属性的上面,名字可随便取,但是括号里面的属性必须是下面定义出来的

常用注解

1.@Documented:指定被修饰的该Annotation可以被javadoc工具提取成文档.

2.@Target:指定被修饰的Annotation可以放置的位置(被修饰的目标)

例如:

@Target(ElementType.TYPE)                   接口、类

@Target(ElementType.FIELD)                     属性
@Target(ElementType.METHOD)                方法
@Target(ElementType.PARAMETER)          方法参数 parameter
@Target(ElementType.CONSTRUCTOR)     构造函数constructor
@Target(ElementType.LOCAL_VARIABLE)   局部变量
@Target(ElementType.ANNOTATION_TYPE)   注解
@Target(ElementType.PACKAGE)                 包 

当一个注解类的@Target()可以为多种位置时,需要在小括号内添加花括号,并且使用逗号隔开

3.@NonNull:属性不为null 

4. @Inherited:指定被修饰的Annotation将具有继承性 

5.@Retention:定义注解的保留策略 有以下三种

 @Retention(RetentionPolicy.SOURCE)             //注解仅存在于源码中,在class字节码文件中不含
 @Retention(RetentionPolicy.CLASS)              //默认的保留策略,注解会在class字节码文件中存在,但运行时无法获得,
 @Retention(RetentionPolicy.RUNTIME)            //注解会在class字节码文件中存在,在运行时可以通过反射获取到

如果 用的是@Retention(RetentionPolicy.CLASS),在TestController类的两个注解只能获取到一个

public class test {
    public static void main(String[] args) {
        for (Annotation a : TestController.class.getAnnotations()) {
            System.out.println(a);
        }
        ;
    }
}

 如果 用的是@Retention(RetentionPolicy.RUNTIME),可获取到两个

value为属性,可在注解类中定义其他属性,在赋值时value属性可以不用定义直接引号赋值即可,但是其他的要明确定义

 获取某注解类的属性值时,需要进行判断

如果要判断在某一个类上是否存在某一注解

MyAnnotation annotation = TestController.class.getAnnotation(MyAnnotation.class);
if(annotation!=null){
    System.out.println(annotation.value());
}

需求:在自定义方法上增加环绕通知

定义一个类,需要的注解:

1.@Aspect:定义切面(需要与@Component一起使用)

2.@Component:定义为组件

3.@Slf4j:日志

在环绕通知的方法上自定义范围

@Pointcut("execution(* com.sjy.controller.*Controller.*(..))")

*表示以 Controller结尾的类的所以方法,可有参也可无参,具体代码如下:

package com.sjy.controller;

import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.Signature;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;
//切面
@Aspect
@Component//组件
@Slf4j
public class LogAop {


    @Pointcut("execution(* com.sjy.controller.*Controller.*(..))")
    public void points(){}

    @Around("points()")
    public Object around(ProceedingJoinPoint point){
        //获得方法名称
        Signature methodName = point.getSignature();
        log.info(methodName+"进来了");
        long l1 = System.currentTimeMillis();
        Object obj=null;
        try {
            //执行方法
            obj=point.proceed(point.getArgs());
        } catch (Throwable throwable) {
            throwable.printStackTrace();
        }
        log.info(methodName+"出来了-"+(System.currentTimeMillis()-l1));
         return obj;
    }
}

但是并不是每个方法都需要被环绕通知的,那就需要不断更改条件,这样是不方便的

我们可以以注解的形式标注在方法上

定义注解类

package com.sjy.annotation;

import java.lang.annotation.*;

/**
 * @author 小宝的宝
 */
@Documented
@Inherited
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface MyLog {

}

在环绕通知的类中给注解类权限

然后只需要在想要环绕的方法上 进行标注注解即可

针对于返回结果的注解及帮助类

1.注解类

package com.sjy.response;

import java.lang.annotation.*;

/**
 * @author hgh
 */
@Retention(value = RetentionPolicy.RUNTIME)
@Documented
@Target({ElementType.METHOD})
public @interface ResponseResult {

}

2.枚举类

package com.sjy.response;

import java.io.Serializable;



public enum ResultCode implements Serializable {

    /* 正常状态 */
    SUCCESS(100, "成功"),
    FAILURE(101, "失败"),
    UNKNOWN(102, "未知响应"),
    /**
     * 用户code范围: 200~300;
     */
    USER_ACCOUNT_NOT_FIND(201, "用户名不存在"),
    USER_ACCOUNT_DISABLED(202, "该用户已被禁用"),
    USER_PASSWORD_NOT_MATCH(203, "该用户密码不一致"),
    USER_PERMISSION_ERROR(204, "该用户不具备访问权限"),
    USER_STATE_OFF_LINE(205, "该用户未登录");

    private final Integer code;
    private final String message;

    ResultCode(Integer code, String message) {
        this.code = code;
        this.message = message;
    }

    public Integer getCode() {
        return code;
    }

    public String getMessage() {
        return message;
    }

    public static ResultCode queryCode(Integer code) {
        for (ResultCode value : values()) {
            if (code.equals(value.code)) {
                return value;
            }
        }
        return UNKNOWN;
    }

}

3.帮助类

package com.sjy.response;

import lombok.Data;

import java.io.Serializable;


@Data
public class Result<T> implements Serializable {

    private final int code;
    private final String message;
    private final T data;

    /**
     * 私有构造, 只允许通过static调用构造
     *
     * @param resultCode 结果枚举
     * @param data       响应数据
     */
    private Result(ResultCode resultCode, T data) {
        this.code = resultCode.getCode();
        this.message = resultCode.getMessage();
        this.data = data;
    }

    /**
     * 成功调用返回的结果(无数据携带)
     *
     * @return Result
     */
    public static Result success() {
        return success(null);
    }

    /**
     * 成功调用返回的结果(数据携带)
     *
     * @return Result
     */
    public static <T> Result success(T data) {
        return new Result(ResultCode.SUCCESS, data);
    }

    /**
     * 失败调用返回的结果(数据携带)
     *
     * @param resultCode 状态枚举
     * @param data       携带的数据
     * @return Result
     */
    public static <T> Result failure(ResultCode resultCode, T data) {
        return new Result(resultCode, data);
    }

    /**
     * 失败调用返回的结果(无数据携带)
     *
     * @param resultCode 状态枚举
     * @return Result
     */
    public static Result failure(ResultCode resultCode) {
        return failure(resultCode, null);
    }

}

4.注解解析类

package com.sjy.response;

import org.springframework.core.MethodParameter;
import org.springframework.http.MediaType;
import org.springframework.http.server.ServerHttpRequest;
import org.springframework.http.server.ServerHttpResponse;
import org.springframework.web.bind.annotation.RestControllerAdvice;
import org.springframework.web.servlet.mvc.method.annotation.ResponseBodyAdvice;


@RestControllerAdvice
public class ResponseParse implements ResponseBodyAdvice {

    @Override
    public boolean supports(MethodParameter methodParameter, Class aClass) {
        //返回值决定他是否需要进入beforeBodyWrite
        return methodParameter.getMethod().isAnnotationPresent(ResponseResult.class);
    }

    @Override
    public Object beforeBodyWrite(Object o, MethodParameter methodParameter, MediaType mediaType, Class aClass, ServerHttpRequest serverHttpRequest, ServerHttpResponse serverHttpResponse) {
        //更改返回值
        if (o == null) {
            return Result.success();
        }
        if (o instanceof Integer) {
            return Result.failure(ResultCode.queryCode((Integer) o));
        }
        if (o instanceof ResultCode) {
            return Result.failure((ResultCode) o);
        }
        if (o instanceof Result) {
            return o;
        }
        return null;
    }

}

  • 3
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值