使用Redis實(shí)現(xiàn)限流

使用Redis實(shí)現(xiàn)限流

  • 原理:使用Redis的Hash數(shù)據(jù)結(jié)構(gòu),把對(duì)應(yīng)的key蘸炸、接口url、限流規(guī)則存放進(jìn)redis尖奔,在攔截器中對(duì)接口進(jìn)行攔截搭儒,獲取到有配置的url,進(jìn)行規(guī)則獲取提茁,獲取到規(guī)則之后淹禾,對(duì)redis對(duì)應(yīng)接口對(duì)應(yīng)key進(jìn)行increment自增的操作(increment 指令是線程安全的,不用擔(dān)心并發(fā)的問題),如果是第一次的話,設(shè)置該key的過期時(shí)間茴扁,過期時(shí)間為配置時(shí)間铃岔,單位為配置單位,下次調(diào)用的時(shí)候峭火,如果當(dāng)前接口對(duì)應(yīng)key的自增數(shù)大于配置的limit數(shù)則進(jìn)行請(qǐng)求超出限制的提示毁习。

具體步驟

  1. 引入Redis依賴包,和其他工具包
        <!-- 阿里json -->
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>fastjson</artifactId>
            <version>1.2.41</version>
        </dependency>
                <!-- 整合Redis start -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-redis</artifactId>
        </dependency>
        <!-- 整合Redis end -->
  1. 在yml文件中編寫限流接口和限流規(guī)則
request_limit:
  # 限流的接口
  url: /utils/redis,/demo/user/account
  rules:
    # 限流規(guī)則智嚷,每秒3次調(diào)用
    limit: 3
    time: 1
    timeUnit: SECONDS
  1. 編寫限流配置類
public class RequestLimitConfig implements Serializable {

    private static final long serialVersionUID = 1101875328323558092L;

    // 最大請(qǐng)求次數(shù)
    private long limit;
    // 時(shí)間
    private long time;
    // 時(shí)間單位
    private TimeUnit timeUnit;

    public RequestLimitConfig() {
        super();
    }

    public RequestLimitConfig(long limit, long time, TimeUnit timeUnit) {
        super();
        this.limit = limit;
        this.time = time;
        this.timeUnit = timeUnit;
    }

    public long getLimit() {
        return limit;
    }

    public void setLimit(long limit) {
        this.limit = limit;
    }

    public long getTime() {
        return time;
    }

    public void setTime(long time) {
        this.time = time;
    }

    public TimeUnit getTimeUnit() {
        return timeUnit;
    }

    public void setTimeUnit(TimeUnit timeUnit) {
        this.timeUnit = timeUnit;
    }

    @Override
    public String toString() {
        return "RequestLimitConfig [limit=" + limit + ", time=" + time + ", timeUnit=" + timeUnit + "]";
    }
}
  1. 繼承HandlerInterceptorAdapter類,實(shí)現(xiàn)其方法編寫限流攔截器
import com.alibaba.fastjson.JSONObject;
import com.example.demo.config.RequestLimitConfig;
import com.example.demo.constants.GlobalConstants;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.handler.HandlerInterceptorAdapter;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.MediaType;
import org.springframework.util.StringUtils;

import java.nio.charset.StandardCharsets;

/**
 *  接口限流攔截器
 */
public class RequestLimitInterceptor extends HandlerInterceptorAdapter {

    private static final Logger log = LoggerFactory.getLogger(RequestLimitInterceptor.class);

    @Autowired
    private RedisTemplate redisTemplate;

    /**
     * 在方法被調(diào)用前執(zhí)行。在該方法中可以做類似校驗(yàn)的功能纺且。如果返回true盏道,則繼續(xù)調(diào)用下一個(gè)攔截器
     *
     * @param request
     * @param response
     * @param handler
     * @return
     * @throws Exception
     */
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        /**
         * 獲取到請(qǐng)求的URI
         */
        String contentPath = request.getContextPath();
        String uri = request.getRequestURI();
        if (!StringUtils.isEmpty(contentPath) && !contentPath.equals("/")) {
            uri = uri.substring(uri.indexOf(contentPath) + contentPath.length());
        }
        log.info("uri={}", uri);

        /**
         * 嘗試從hash中讀取得到當(dāng)前接口的限流配置
         */
        String str = this.redisTemplate.opsForHash().get(GlobalConstants.REQUEST_LIMIT_CONFIG, uri).toString();
        RequestLimitConfig requestLimitConfig = JSONObject.parseObject(str, RequestLimitConfig.class);
        if (requestLimitConfig == null) {
            log.info("該uri={}沒有限流配置", uri);
            return true;
        }

        String limitKey = GlobalConstants.REQUEST_LIMIT + ":" + uri;

        /**
         * 當(dāng)前接口的訪問次數(shù) +1 increment 指令是線程安全的,不用擔(dān)心并發(fā)的問題
         */
        long count = this.redisTemplate.opsForValue().increment(limitKey);
        if (count == 1) {
            /**
             * 第一次請(qǐng)求载碌,設(shè)置key的過期時(shí)間
             */
            this.redisTemplate.expire(limitKey, requestLimitConfig.getTime(), requestLimitConfig.getTimeUnit());
            log.info("設(shè)置過期時(shí)間:time={}, timeUnit={}", requestLimitConfig.getTime(), requestLimitConfig.getTimeUnit());
        }

        log.info("請(qǐng)求限制猜嘱。limit={}, count={}", requestLimitConfig.getLimit(), count);

        if (count > requestLimitConfig.getLimit()) {
            /**
             * 限定時(shí)間內(nèi),請(qǐng)求超出限制嫁艇,響應(yīng)客戶端錯(cuò)誤信息泉坐。
             */
            response.setContentType(MediaType.TEXT_PLAIN_VALUE);
            response.setCharacterEncoding(StandardCharsets.UTF_8.name());
            response.getWriter().write("服務(wù)器繁忙,稍后再試");
            return false;
        }
        return true;
    }

    /**
     * 在方法執(zhí)行后調(diào)用(暫未使用)
     *
     * @param request
     * @param response
     * @param handler
     * @param modelAndView
     * @throws Exception
     */
    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        super.postHandle(request, response, handler, modelAndView);
    }
}
  1. 實(shí)現(xiàn)WebMvcConfigurer接口,編寫資源配置類
/**
 * 資源配置器
 */
@Configuration
public class WebMvcConfiguration implements WebMvcConfigurer {

   @Value("${request_limit.url}")
    private String url;

    // 添加攔截器
    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        // 添加限流的接口
        registry.addInterceptor(this.requestLimitInterceptor())
                .addPathPatterns(url.split(","));
    }

    @Bean
    public RequestLimitInterceptor requestLimitInterceptor() {
        return new RequestLimitInterceptor();
    }
}
  1. 編寫服務(wù)啟動(dòng)時(shí)注入限流接口和規(guī)則
import com.alibaba.fastjson.JSONObject;
import com.example.demo.constants.GlobalConstants;
import com.example.demo.exception.BusinessException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.ApplicationArguments;
import org.springframework.boot.ApplicationRunner;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Component;

/**
 * 初始化運(yùn)行方法
 */
@Component
public class ApplicationRunnerImpl implements ApplicationRunner {

    private static final Logger log = LoggerFactory.getLogger(ApplicationRunnerImpl.class);

    // 引入redis
    @Autowired
    private RedisTemplate redisTemplate;

    @Value("${request_limit.url}")
    private String url;

    @Value("${request_limit.rules.limit}")
    private String limit;
    @Value("${request_limit.rules.time}")
    private String time;
    @Value("${request_limit.rules.timeUnit}")
    private String timeUnit;

    @Override
    public void run(ApplicationArguments args) throws Exception {
        // 已經(jīng)寫好了方法
        //TestCron.init();

        JSONObject rulesJson = new JSONObject();
        rulesJson.put("limit", limit);
        rulesJson.put("time", time);
        rulesJson.put("timeUnit", timeUnit);

        try {
            String[] urlArr = url.split(",");
            // 初始化在Redis中存入接口限流規(guī)則
            for (String item : urlArr) {
                redisTemplate.opsForHash().put(GlobalConstants.REQUEST_LIMIT_CONFIG, item, rulesJson);
                log.info("Redis存放成功,接口地址:" + item + " 限流規(guī)則:" + "  時(shí)間:" + time + "  單位:" + timeUnit + "  次數(shù):" + limit);
            }
        } catch (Exception e) {
            e.printStackTrace();
            throw new BusinessException("限流規(guī)則配置錯(cuò)誤,請(qǐng)使用逗號(hào)分割");
        }
    }
}
  1. Controller接口寫入配置的路徑即可
  2. 進(jìn)行請(qǐng)求,快速刷新瀏覽器裳仆,結(jié)果如圖
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末腕让,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子歧斟,更是在濱河造成了極大的恐慌纯丸,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,214評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件静袖,死亡現(xiàn)場(chǎng)離奇詭異觉鼻,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)队橙,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,307評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門坠陈,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人捐康,你說我怎么就攤上這事仇矾。” “怎么了解总?”我有些...
    開封第一講書人閱讀 152,543評(píng)論 0 341
  • 文/不壞的土叔 我叫張陵贮匕,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我花枫,道長(zhǎng)刻盐,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,221評(píng)論 1 279
  • 正文 為了忘掉前任劳翰,我火速辦了婚禮敦锌,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘佳簸。我一直安慰自己乙墙,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,224評(píng)論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著伶丐,像睡著了一般悼做。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上哗魂,一...
    開封第一講書人閱讀 49,007評(píng)論 1 284
  • 那天肛走,我揣著相機(jī)與錄音,去河邊找鬼录别。 笑死朽色,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的组题。 我是一名探鬼主播葫男,決...
    沈念sama閱讀 38,313評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼崔列!你這毒婦竟也來了梢褐?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 36,956評(píng)論 0 259
  • 序言:老撾萬榮一對(duì)情侶失蹤赵讯,失蹤者是張志新(化名)和其女友劉穎盈咳,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體边翼,經(jīng)...
    沈念sama閱讀 43,441評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡鱼响,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,925評(píng)論 2 323
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了组底。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片丈积。...
    茶點(diǎn)故事閱讀 38,018評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖债鸡,靈堂內(nèi)的尸體忽然破棺而出江滨,到底是詐尸還是另有隱情,我是刑警寧澤娘锁,帶...
    沈念sama閱讀 33,685評(píng)論 4 322
  • 正文 年R本政府宣布牙寞,位于F島的核電站,受9級(jí)特大地震影響莫秆,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜悔详,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,234評(píng)論 3 307
  • 文/蒙蒙 一镊屎、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧茄螃,春花似錦缝驳、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,240評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽运怖。三九已至,卻和暖如春夏伊,著一層夾襖步出監(jiān)牢的瞬間摇展,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,464評(píng)論 1 261
  • 我被黑心中介騙來泰國打工溺忧, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留咏连,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,467評(píng)論 2 352
  • 正文 我出身青樓鲁森,卻偏偏與公主長(zhǎng)得像祟滴,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子歌溉,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,762評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容