当前位置: 首页 > news >正文

Aop切面技术之存储用户信息

一、背景

在我们日常项目中,常常会将用户的认证信息存入缓存中,方便我们在程序执行中,进行获取用户信息。本篇文章主要是介绍使用自定义注解和AOP切面技术进行实现,这也也是非常容易使用的。

二、代码详解

2.1 自定义注解

定义一个注解,用于标记需要自动注入当前用户信息的方法。

import java.lang.annotation.ElementType;  
import java.lang.annotation.Retention;  
import java.lang.annotation.RetentionPolicy;  
import java.lang.annotation.Target;  @Target(ElementType.METHOD)  
@Retention(RetentionPolicy.RUNTIME)  
public @interface InjectCurrentUser {  
}
2.2 创建切面

创建一个切面来拦截所有标记了@InjectCurrentUser注解的方法,并在调用它们之前获取当前用户信息,然后作为参数传递给这些方法(注意:Java的方法签名在编译时是固定的,不能直接向现有方法添加参数。因此,这里假设我们修改方法签名以接受用户信息作为参数,或者使用其他方式如ThreadLocal)。

@Aspect  
@Component  
public class UserAspect {  @Before("@annotation(InjectCurrentUser)")  public void beforeMethod(JoinPoint joinPoint) {  Authentication authentication = SecurityContextHolder.getContext().getAuthentication();  if (authentication != null && authentication.getPrincipal() instanceof UserDetails) {  UserDetails userDetails = (UserDetails) authentication.getPrincipal();  UserContextHolder.setCurrentUser(userDetails);  // 如果需要,可以在这里添加日志或其他逻辑  System.out.println("Setting current user: " + userDetails.getUsername());  }  }  // 可以添加一个@After或@AfterReturning切面来清除ThreadLocal中的用户信息  // 这样做是为了避免内存泄漏,尤其是在长时间运行的线程或线程池中  @After("@annotation(InjectCurrentUser)")  public void afterMethod(JoinPoint joinPoint) {  UserContextHolder.clearCurrentUser();  }  
}
2.3 创建ThreadLocal存储用户信息

定义一个ThreadLocal来存储当前用户的信息(比如UserDetails

public class UserContextHolder {  private static final ThreadLocal<UserDetails> currentUser = new ThreadLocal<>();  public static void setCurrentUser(UserDetails userDetails) {  currentUser.set(userDetails);  }  public static UserDetails getCurrentUser() {  return currentUser.get();  }  // 清除ThreadLocal中的用户信息,避免内存泄漏  public static void clearCurrentUser() {  currentUser.remove();  }  
}
2.4 业务逻辑中获取用户信息

业务逻辑中,通过UserContextHolder.getCurrentUser()来获取当前用户的信息

public class SomeService {  public void someBusinessMethod() {  UserDetails user = UserContextHolder.getCurrentUser();  if (user != null) {  // 使用用户信息进行业务逻辑处理  System.out.println("Doing something with user: " + user.getUsername());  }  }  
}

三、总结

内存泄漏ThreadLocal可能会导致内存泄漏,特别是当使用线程池时,因为线程可能会被重用,而ThreadLocal中的值可能不会被自动清除。因此,在不再需要时显式清除ThreadLocal中的值是一个好习惯。
依赖注入:尽管在这个例子中我们使用了AOP来设置ThreadLocal,但在某些情况下,你可能还想通过依赖注入来传递用户信息,特别是当你需要在多个组件或服务之间共享用户信息时。然而,对于跨线程或跨方法调用的情况,ThreadLocal通常是一个更好的选择。

http://www.lryc.cn/news/426327.html

相关文章:

  • FreeBSD 针对OpenSSH 高危漏洞发布紧急补丁
  • 【C语言小项目】五子棋游戏
  • 基于Java语言的能源管理系统-水电气热油数据采集系统
  • 人工智能在肿瘤亚型分类领域的研究进展|顶刊速递·24-08-13
  • Taro+Vue 创建微信小程序
  • 智能安全守护,寺庙安全用电解决方案
  • 加热系统加入达温即停和保温功能
  • C++_2_ inline内联函数 宏函数(2/3)
  • ROS执行多个节点报错(遥控运动及SLAM建图)
  • Spring Boot项目中实现文件的上传、下载和预览功能
  • 【JAVA入门】Day21 - 时间类
  • SQL server数据库备份和还原
  • B站搜索建库架构优化实践
  • XSS反射实战
  • 远程消息传递的艺术:NSDistantObject在Objective-C中的妙用
  • 指向派生类的基类指针、强转为 void* 再转为基类指针、此时调用虚函数会发生什么?
  • 操作系统(Linux实战)-进程创建、同步与锁、通信、调度算法-学习笔记
  • react的setState中为什么不能用++?
  • 2.2算法的时间复杂度与空间复杂度——经典OJ
  • 【CentOS 】DHCP 更改为静态 IP 地址并且遇到无法联网
  • Linux 操作系统 --- 信号
  • 黑马前端——days09_css
  • 【Python爬虫】技术深度探索与实践
  • 智启万象|挖掘广告变现潜力,保障支付安全便捷
  • 函数递归,匿名、内置行数,模块和包,开发规范
  • Springboot3 整合swagger
  • 查看同一网段内所有设备的ip
  • Spark MLlib 特征工程(上)
  • 《SPSS零基础入门教程》学习笔记——03.变量的统计描述
  • 2024年杭州市网络与信息安全管理员(网络安全管理员)职业技能竞赛的通知