springboot攔截器無法註入redisTemplate的解決方法

在工作中我們經常需要做登錄攔截驗證或者其他攔截認證功能,基於springboot項目下我們很容易想到結合redis做的分佈式攔截,把用戶登錄或者需要驗證的信息放到redis裡面。但是在寫攔截器的時候發現redisTemplate一直無法註入進來,最後查資料才發現springboot攔截器是在Bean實例化之前執行的,所以Bean實例無法註入。

先看下問題,新建一個攔截器,然後註入redisTemplate

/**
 * @author: lockie
 * @Date: 2019/8/13 16:16
 * @Description: 接口限流攔截器
 */
@Component
public class AccessLimitIntercept implements HandlerInterceptor {
    private static final Logger logger = LoggerFactory.getLogger(AccessLimitIntercept.class);
 
    @Autowired
    private StringRedisTemplate redisTemplate;
 
    /**
     * 接口調用前檢查對方ip是否頻繁調用接口
     * @param request
     * @param response
     * @param handler
     * @return
     * @throws Exception
     */
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        try {
 
            String test = redisTemplate.opsForValue().get("test");
            logger.info(test);
        } catch (Exception e) {
            logger.error("API請求限流攔截異常,異常原因:", e);
            throw new ParameterException(e);
        }
        return true;
    }
 
    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
 
    }
 
    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
 
    }
 
    private boolean setResponse(Results results, HttpServletResponse response) throws IOException {
        ServletOutputStream outputStream = null;
        try {
            response.setHeader("Content-type", "application/json; charset=utf-8");
            outputStream = response.getOutputStream();
            outputStream.write(JsonUtil.toJson(results).getBytes("UTF-8"));
        } catch (Exception e) {
            logger.error("setResponse方法報錯", e);
            return false;
        } finally {
            if (outputStream != null) {
                outputStream.flush();
                outputStream.close();
            }
        }
        return true;
    }
}

然後配置攔截器,新建一個攔截器config類,我們這裡讓所有的url都攔截

@Configuration
public class WebFilterConfig implements WebMvcConfigurer {
 
	/**
	 * 多個攔截器組成一個攔截器鏈
	 * @param registry
	 */
	@Override
	public void addInterceptors(InterceptorRegistry registry) {
		registry.addInterceptor(new AccessLimitIntercept()).addPathPatterns("/**");
	}
 
}

新建一個controller裡面增加一個接口,使用postman調用接口

@RestController
@RequestMapping("/")
public class PingController extends BaseController {
 
	@GetMapping(value = "/ping")
	public Results ping() {
		return succeed("pong", "");
	}
 
}

使用postman調用/ping接口,發現返回系統錯誤

控制臺發現是空指針的錯誤,定位錯誤發現當我們想獲取redis裡面key是test的值時,發現redisTemplate沒有是空的沒有註入進來

解決問題:

知道攔截器執行在bean實例化前執行的,那麼我們就讓攔截器執行的時候實例化攔截器Bean,在攔截器配置類裡面先實例化攔截器,然後再獲取

@Configuration
public class WebFilterConfig implements WebMvcConfigurer {
 
	/**
	 * 這裡需要先將限流攔截器入住,不然無法獲取到攔截器中的redistemplate
	 * @return
	 */
	@Bean
	public AccessLimitIntercept getAccessLimitIntercept() {
		return new AccessLimitIntercept();
	}
 
	/**
	 * 多個攔截器組成一個攔截器鏈
	 * @param registry
	 */
	@Override
	public void addInterceptors(InterceptorRegistry registry) {
		registry.addInterceptor(getAccessLimitIntercept()).addPathPatterns("/**");
	}
 
 
}

使用postman再次請求就可以訪問到

到此這篇關於springboot攔截器無法註入redisTemplate的解決方法的文章就介紹到這瞭,更多相關springboot攔截器無法註入redisTemplate內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: