浅谈基于SpringBoot实现一个简单的权限控制注解

作者:ly931003 时间:2022-01-23 16:42:47 

 注解是 JDK 5.0 引入的一种注释机制。注解可以作用在类型(类、接口、枚举等)、属性、方法、参数等不同位置,具体的 JDK 版本所支持的注解位置可参考 java.lang.annotation.ElementType 。此外还有注解的策略,也就是 RetentionPolicy ,这里不加赘述。

注解可以实现很多功能,其中最主要的就是进行代码标注,所以有时候注解也叫做标注。使用起来也基本顾名思义,就是对代码进行标注,简化部分代码的逻辑。

下面我们就着手实现一个简单的权限控制注解,来对注解有一个基本的了解。

准备

@Permission 注解

注解本身的代码很简单。下面实现的是一个 @Permission 注解,为了方便使用,这里只提供一个属性value,因为如果一个注解中有一个名称为value的属性,且你只想设置value属性(即其他属性都采用默认值或者你只有一个value属性),那么可以省略掉“value=”部分。


import java.lang.annotation.*;

@Target({ElementType.PARAMETER}) // 注解可用于参数
@Retention(RetentionPolicy.RUNTIME) // 注解在运行时可由JVM读入
@Documented
public @interface Permission {
 String value() default "";
}

User 类

一个简单的User类,包含 permissions 用于保存用户的权限。


import lombok.Data;

@Data
public class User {
 private String id;
 private String name;

private Set<String> permissions;
}

UserService 类

简单的 Service 类,用于判断权限。


@Service
public class UserService {
 public boolean checkCreatePermission(@Permission("创建用户") User user) {
   return true;
 }

public boolean checkDeletePermission(@Permission("删除用户") User user) {
   return true;
 }
}

PermissionAspect 类

利用 SpringBoot 简单地设置切面,获取注解并使用。这里直接


@Aspect
@Component
public class PermissionAspect {
 // 需要修改为实际的 Service 所在的 Package
 @Pointcut("execution(public * tk.yubarimelon.MongoDemo.service.*.*(..))")
 public void permissionCheck() {
 }

@Around("permissionCheck()")
 public Object before(ProceedingJoinPoint joinPoint) throws Throwable {
   Object[] params = joinPoint.getArgs();
   // 获取方法,此处可将signature强转为MethodSignature
   MethodSignature signature = (MethodSignature) joinPoint.getSignature();
   Method method = signature.getMethod();

// 获取参数注解,1维是参数,2维是注解
   Annotation[][] parameterAnnotations = method.getParameterAnnotations();
   for (int i = 0; i < parameterAnnotations.length; i++) {
     Object param = params[i];
     Annotation[] annotations = parameterAnnotations[i];
     if (!(param instanceof User) || annotations.length == 0) {
       continue;
     }
     for (Annotation annotation : annotations) {
       if (annotation.annotationType().equals(Permission.class)) {
         Permission permission = (Permission) annotation;
         User user = (User) param;
         if (CollectionUtils.isEmpty(user.getPermissions())) {
           log.error(user.getName() + " 无任何权限!");
           return false;
         }
         if (!StringUtils.hasLength(permission.value())) {
           log.error(joinPoint.getSignature().toString() + "权限设置异常");
           return false;
         }
         if (!user.getPermissions().contains(permission.value())) {
           log.error(joinPoint.getSignature().toString() +": "+user.getName() + " 无权限: " + permission.value());
           return false;
         }
         return joinPoint.proceed();
       }
     }
   }
   return joinPoint.proceed();
 }
}

ApplicationTests 类

简单的测试类,用于测试代码。这里简单的配置一个用户只有创建用户的权限


@SpringBootTest
class ApplicationTests {

@Autowired
 UserService userService;

@Test
 void contextLoads() {
 }

@Test
 void checkUser() {
   User user = new User();
   user.setName("小明");
   Set<String> permissions = new HashSet<>();
   permissions.add("创建用户");
   user.setPermissions(permissions);

System.out.println("checkCreatePermission " + userService.checkCreatePermission(user));
   System.out.println("checkDeletePermission " + userService.checkDeletePermission(user));
 }
}

输出如下日志,证明权限设置起作用了。

checkCreatePermission true
2021-01-31 11:44:45.895 ERROR 12388 --- [           main] t.y.MongoDemo.aop.PermissionAspect       : boolean tk.yubarimelon.MongoDemo.service.UserService.checkDeletePermission(User): 小明 无权限: 删除用户
checkDeletePermission false

来源:https://segmentfault.com/a/1190000039134345

标签:SpringBoot,权限控制,注解
0
投稿

猜你喜欢

  • c#实现输出本月的月历

    2023-07-18 19:41:33
  • 简单实现安卓里百度地图持续定位

    2023-07-29 07:59:22
  • SpringBoot 整合mapstruct的实现步骤

    2023-11-28 00:51:00
  • IDEA如何自动生成serialVersionUID的设置

    2023-08-09 04:00:33
  • Java的Struts框架简介与环境配置教程

    2023-10-29 05:23:03
  • 解决idea爆红 cant resolve symbol String的问题解析

    2023-10-11 10:47:30
  • IDEA教程之Activiti插件图文详解

    2023-11-14 23:06:45
  • 如何让java只根据数据库表名自动生成实体类

    2022-02-24 04:25:52
  • Java实现几种序列化方式总结

    2023-02-13 06:18:27
  • java 创建线程的四种方式

    2023-11-02 21:38:07
  • JVM教程之内存管理和垃圾回收(三)

    2023-11-10 15:49:54
  • 你所不知道的Spring自动注入详解

    2021-09-04 19:30:08
  • mybatis-plus中lambdaQuery()与lambdaUpdate()比较常见的使用方法总结

    2023-11-24 22:43:04
  • Linux下Hadoop 2.7.3 安装搭建过程

    2023-11-08 05:13:59
  • flutter窗口初始和绘制流程详析

    2023-08-17 21:07:30
  • Android6.0 消息机制原理解析

    2023-08-06 12:19:44
  • Java里的static在Kotlin里如何实现

    2023-08-24 20:25:33
  • Java多线程Thread类的使用详解

    2023-11-11 13:08:11
  • 关于spring的自定义缓存注解分析

    2023-11-28 17:02:50
  • springBoot之如何获取接口请求数据和返回数据实现日志

    2023-11-23 10:43:58
  • asp之家 软件编程 m.aspxhome.com