欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

springboot獲取當(dāng)前用戶信息的三種方式

 更新時(shí)間:2024年10月16日 11:10:40   作者:何中應(yīng)  
本文詳細(xì)介紹了在開發(fā)中獲取當(dāng)前操作用戶信息的三種方式:使用ThreadLocal存儲用戶信息、通過攔截器和注解結(jié)合Shiro框架獲取用戶信息、以及使用Redis存儲用戶信息,每種方式都有具體的實(shí)現(xiàn)步驟和注意事項(xiàng),可根據(jù)實(shí)際需求選擇合適的方法

說明:在開發(fā)中,我們經(jīng)常需要獲取當(dāng)前操作的用戶信息,如創(chuàng)建用戶、創(chuàng)建訂單時(shí),我們需要記錄下創(chuàng)建人,本文介紹獲取當(dāng)前用戶信息的三種方式。

方式一:使用ThreadLocal

ThreadLocal本質(zhì)上是一個(gè)Map,鍵是當(dāng)前線程,值是存入的信息。我們可以在用戶登錄,校驗(yàn)用戶信息后,將所需要的用戶信息存入到ThreadLocal中,如用戶ID、用戶Token等,然后在需要的時(shí)候直接使用即可。

如下,在preHandle()方法中,將當(dāng)前用戶的ID存入到TokenThreadLocal對象中,

@Component
public class TokenInterceptor implements HandlerInterceptor {

    @Value("${token.key}")
    private String tokenKey;

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
    
        // 獲取請求頭 拿到Token
        String token = request.getHeader("Token");
        String authToken = request.getHeader("authentication");

        if (StrUtil.isAllEmpty(token,authToken)){
           return responseHandler(response);
        }

        //  校驗(yàn)Token的合法性和有效性
        JWT jwt = JWTUtil.parseToken(StrUtil.isBlank(token) ? authToken : token);

        try {
            // 校驗(yàn)Token是否合法 校驗(yàn)Token是否過期
            if (!(jwt.setKey(tokenKey.getBytes()).verify() && jwt.validate(0))){
                return responseHandler(response);
            }
        } catch (Exception e) {
            // 拋出自定義異常 Token是非法的
            // throw new RuntimeException(e);
            return responseHandler(response);
        }

        // 把Token的信息解析出來放到ThreadLocal
        Long id = Convert.toLong(jwt.getPayload("id"));

        // 設(shè)置本地線程池中的用戶ID
        TokenThreadLocal.set(id);
        
        // 放行
        return true;
    }

本地線程對象,TokenThreadLocal

/**
 * 本地線程對象
 *
 * 存放用戶ID
 */
public class TokenThreadLocal {

    /**
     * 創(chuàng)建一個(gè)ThreadLocal對象
     */
    private static final ThreadLocal<Long> THREAD_LOCAL= new ThreadLocal<>();

    /**
     * 添加一個(gè)數(shù)據(jù)
     * @param key
     */
    public static void set(Long key){
        THREAD_LOCAL.set(key);
    }

    /**
     * 獲取一個(gè)數(shù)據(jù)
     * @return
     */
    public static Long get(){
        return THREAD_LOCAL.get();
    }

    /**
     * 刪除一個(gè)數(shù)據(jù)
     */
    public static void remove(){
        THREAD_LOCAL.remove();
    }
}

需要的時(shí)候,直接調(diào)用其get()方法,下面是使用AOP+自定義注解實(shí)現(xiàn)對創(chuàng)建、更新操作字段的填充;

在這里插入圖片描述

注意,需要在afterCompletion()方法中調(diào)用ThreadLocal的remove()方法,避免內(nèi)存泄漏;

在這里插入圖片描述

方式二:通過攔截器和相應(yīng)注解實(shí)現(xiàn)

如果項(xiàng)目中,登錄校驗(yàn)框架使用的是Shiro,有一種更方便的方式,如下:

第一步:創(chuàng)建一個(gè)自定義注解,如LoginInfo,表示登錄用戶的信息,注意元注解的屬性;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

/**
 * 登錄用戶
 * @author 
 */
@Target(value = ElementType.PARAMETER)
@Retention(RetentionPolicy.RUNTIME)
public @interface LoginInfo {
}

第二步:創(chuàng)建MVC的配置類,注入一個(gè)在線用戶解析對象,后面實(shí)現(xiàn);

import org.decent.modules.integral.resolver.LoginUserArgumentResolver;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.method.support.HandlerMethodArgumentResolver;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
import javax.annotation.Resource;
import java.util.List;

/**
 * web設(shè)置
 * @author
 */
@Configuration
public class WebConfig implements WebMvcConfigurer {

    @Resource
    private LoginUserArgumentResolver loginUserArgumentResolver;

    @Override
    public void addArgumentResolvers(List<HandlerMethodArgumentResolver> resolvers) {
        resolvers.add(loginUserArgumentResolver);
    }
}

第三步:創(chuàng)建在線用戶解析類,其中獲取當(dāng)前用戶的信息使用的是Shiro框架的方法,SecurityUtils.getSubject().getPrincipal(),該方法返回的是一個(gè)Object類型的對象;

import org.apache.shiro.SecurityUtils;
import org.decent.modules.integral.annotation.LoginInfo;
import org.springframework.core.MethodParameter;
import org.springframework.stereotype.Component;
import org.springframework.web.bind.support.WebDataBinderFactory;
import org.springframework.web.context.request.NativeWebRequest;
import org.springframework.web.method.support.HandlerMethodArgumentResolver;
import org.springframework.web.method.support.ModelAndViewContainer;

/**
 * 登錄解析實(shí)現(xiàn)
 *
 * @author
 */
@Component
public class LoginUserArgumentResolver implements HandlerMethodArgumentResolver {
    @Override
    public boolean supportsParameter(MethodParameter methodParameter) {
        return methodParameter.hasParameterAnnotation(LoginInfo.class);
    }

    @Override
    public Object resolveArgument(MethodParameter methodParameter, ModelAndViewContainer modelAndViewContainer, NativeWebRequest nativeWebRequest, WebDataBinderFactory webDataBinderFactory) throws Exception {
        return SecurityUtils.getSubject().getPrincipal();
    }
}

第四步:創(chuàng)建一個(gè)在線用戶的JavaBean,存放一些可能會用得上的屬性;

import lombok.Data;

/**
 * 在線用戶信息
 */
@Data
public class LoginUser {

    /**
     * 登錄人id
     */
    private String id;

    /**
     * 登錄人賬號
     */
    private String username;

    /**
     * 登錄人名字
     */
    private String realname;
}

第五步:在需要使用的接口上,直接使用注解即可。當(dāng)請求訪問該接口時(shí),會被前面的攔截器攔截住,然后把當(dāng)前用戶的信息取出來并封裝到JavaBean對象中,非常方便;

	@PostMapping(value = "/add")
	public Result<?> add(@LoginInfo LoginUser loginUser) {
        ......
	}

方式三:使用Redis存儲用戶信息

這種方式思路和第一種相同,當(dāng)用戶通過校驗(yàn)時(shí),將用戶信息查詢出來并存起來,需要的時(shí)候再取出來用。當(dāng)然,使用Redis存儲比ThreadLocal更靈活一點(diǎn),可以設(shè)置有效時(shí)間。實(shí)現(xiàn)如下:

第一步:登錄驗(yàn)證通過,將用戶信息存入Redis;

    @PostMapping("/login")
    public Result<?> counterLogin(@RequestBody LoginBody LoginUser){
        // 登錄
        LoginUser userInfo = sysLoginService.login(LoginUser.getUsername(), LoginUser.getPassword(),LoginUser.getCounterType());
        
        // 創(chuàng)建Token并返回
        return Result.success(tokenService.createToken(userInfo));
    }
     @Autowired
    private RedisService redisService;

	// 定義有效時(shí)間,為720 * 60 秒,即12小時(shí)
    private final static long EXPIRE_TIME = 720 * 60;

	/**
     * 創(chuàng)建令牌
     */
    public Map<String, Object> createToken(LoginUser loginUser){
        // 生成token
        String token = IdUtils.fastUUID();
        loginUser.setToken(token);
        loginUser.setUserid(loginUser.getSysUser().getUserId());
        loginUser.setUsername(loginUser.getSysUser().getUserName());

        // 保存用戶token
        Map<String, Object> map = new HashMap<String, Object>();
        map.put("token", token);
        map.put("loginUser",loginUser);

		// 將該用戶的信息存入到Redis中
        redisService.setCacheObject(token, loginUser, EXPIRE_TIME, TimeUnit.SECONDS);
        
        return map;
    }

RedisService類相關(guān)方法

/**
 * spring redis 工具類
 **/
@Component
public class RedisService{

    @Autowired
    public RedisTemplate redisTemplate;
    
    /**
     * 緩存基本的對象
     */
    public <T> void setCacheObject(final String key, final T value, final Long timeout, final TimeUnit timeUnit){
        redisTemplate.opsForValue().set(key, value, timeout, timeUnit);
    }

    /**
     * 獲得緩存的基本對象
     */
    public <T> T getCacheObject(final String key){
        ValueOperations<String, T> operation = redisTemplate.opsForValue();
        return operation.get(key);
    }
}

第三步:需要時(shí),根據(jù)當(dāng)前用戶的Token,去Redis中取出該用戶的信息;

    /**
     * 根據(jù)用戶Token獲取用戶身份信息
     *
     * @return 用戶信息
     */
    public LoginUser getLoginUser(String token){
        if (StringUtils.isNotEmpty(token)){
            String userKey = getTokenKey(token);
            LoginUser user = redisService.getCacheObject(userKey);
            return user;
        }
        return null;
    }

用戶的Token是需要放在Request對象里面的,所以可以再寫一個(gè)TokenService對象,用來獲取當(dāng)前用戶的Token,并調(diào)用RedisService獲取當(dāng)前用戶信息,進(jìn)行進(jìn)一步的封裝。

總結(jié)

以上是三種獲取當(dāng)前用戶信息的方式,可以根據(jù)實(shí)際情況選擇;更多相關(guān)springboot獲取當(dāng)前用戶信息內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • spring中使用@Autowired注解無法注入的情況及解決

    spring中使用@Autowired注解無法注入的情況及解決

    這篇文章主要介紹了spring中使用@Autowired注解無法注入的情況及解決,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-09-09
  • Spring-Bean創(chuàng)建對象的步驟方式詳解

    Spring-Bean創(chuàng)建對象的步驟方式詳解

    在本篇文章里小編給大家分享的是關(guān)于Spring-Bean創(chuàng)建對象的步驟方式詳解內(nèi)容,有興趣的朋友們跟著學(xué)習(xí)下。
    2020-02-02
  • springboot jackson自定義序列化和反序列化實(shí)例

    springboot jackson自定義序列化和反序列化實(shí)例

    這篇文章主要介紹了spring boot jackson自定義序列化和反序列化實(shí)例,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-10-10
  • Springcloud sentinel安裝和使用方法解析

    Springcloud sentinel安裝和使用方法解析

    這篇文章主要介紹了Springcloud sentinel安裝和使用方法解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-12-12
  • Java利用 Exchanger 實(shí)現(xiàn)游戲中交換裝備

    Java利用 Exchanger 實(shí)現(xiàn)游戲中交換裝備

    JDK 1.5 開始 JUC 包下提供的 Exchanger 類可用于兩個(gè)線程之間交換信息。下面我們就來看看Java是如何利用Exchanger一行代碼實(shí)現(xiàn)游戲中交換裝備的
    2021-09-09
  • springboot中的@value取不到正確的值問題

    springboot中的@value取不到正確的值問題

    這篇文章主要介紹了springboot中的@value取不到正確的值問題,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-12-12
  • Spring循環(huán)依賴之問題復(fù)現(xiàn)詳解

    Spring循環(huán)依賴之問題復(fù)現(xiàn)詳解

    這篇文章主要為大家詳細(xì)介紹了Spring的循環(huán)依賴什么時(shí)候會出現(xiàn)以及如何解決循環(huán)依賴,文中的示例代碼講解詳細(xì),感興趣的可以學(xué)習(xí)一下
    2022-07-07
  • JVM進(jìn)階教程之字段訪問優(yōu)化淺析

    JVM進(jìn)階教程之字段訪問優(yōu)化淺析

    這篇文章主要給大家介紹了關(guān)于JVM進(jìn)階教程之字段訪問優(yōu)化的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-01-01
  • Java使用Ajax實(shí)現(xiàn)跨域上傳圖片功能

    Java使用Ajax實(shí)現(xiàn)跨域上傳圖片功能

    這篇文章主要介紹了Java使用Ajax實(shí)現(xiàn)跨域上傳圖片功能,需要的朋友可以參考下
    2017-09-09
  • 詳解Spring Cloud 跨服務(wù)數(shù)據(jù)聚合框架

    詳解Spring Cloud 跨服務(wù)數(shù)據(jù)聚合框架

    這篇文章主要介紹了詳解Spring Cloud 跨服務(wù)數(shù)據(jù)聚合框架,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2018-03-03

最新評論