threadlocal:线程域对象。在业务当中,每一个请求到达微服务,都会是一个独立的线程。
如果没有用threadlocal,而是直接把用户保存到本地变量,就可能出现多线程并发修改的安全问题。
而threadlocal会将数据保存到每一个线程的内部,在线程内部创建一个threadlocalMap去保存,从而
使每个线程都有自己独立的存储空间,从而每一个请求发起后都能有自己的空间,相互之间不干扰。
-
校验登录状态
-
请求并携带cookie,从session中获取用户,若用户存在
-
登录账号后,将用户信息保存到threadlocal中(threadlocal相互不影响,如果把登录后把用户信息保存到session中,那么再登录一个账号就会使session改变,引起线程并发安全问题)
-
自定义元数据对象处理器(实现MetaObjectHandler)
-
创建基于threadlocal的封装工具类,用于保存和获取当前登录用户id
-
(在不能获取session的类中,通过封装工具类获取id)
public class BaseContext {
private static ThreadLocal<Long> threadLocal=new ThreadLocal<>();
public static void setCurrentId(Long id){
threadLocal.set(id);
}
public static Long getCurrentId(){
return threadLocal.get();
}
}
-
通过threadlocal封装工具类获取id,自动填充入数据库
@Slf4j
@Component
public class MyMetaObjectHandler implements MetaObjectHandler {
/*插入操作自动填充*/
@Override
public void insertFill(MetaObject metaObject) {
log.info("公共字段自动填充[insert]...");
log.info(metaObject.toString());
metaObject.setValue("createTime", LocalDateTime.now());
metaObject.setValue("updateTime", LocalDateTime.now());
metaObject.setValue("createUser", BaseContext.getCurrentId());
metaObject.setValue("updateUser", BaseContext.getCurrentId());
}
/*更新操作自动填充*/
@Override
public void updateFill(MetaObject metaObject) {
log.info("公共字段自动填充[update]...");
log.info(metaObject.toString());
long id = Thread.currentThread().getId();
log.info("线程id为{}",id);
//StringBuilder stringBuilder=new StringBuilder("");
metaObject.setValue("updateTime", LocalDateTime.now());
metaObject.setValue("updateUser", BaseContext.getCurrentId());
}
}
//实体类中属性
//创建时间
@TableField(fill = FieldFill.INSERT)
private LocalDateTime createTime;
//更新时间
@TableField(fill = FieldFill.INSERT_UPDATE)
private LocalDateTime updateTime;
//创建人
@TableField(fill = FieldFill.INSERT)
private Long createUser;
//修改人
@TableField(fill = FieldFill.INSERT_UPDATE)
private Long updateUser;