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

Redis?Lua腳本實(shí)現(xiàn)ip限流示例

 更新時(shí)間:2022年07月15日 11:45:22   作者:yin  
這篇文章主要介紹了Redis?Lua腳本實(shí)現(xiàn)ip限流示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

引言

分布式限流最關(guān)鍵的是要將限流服務(wù)做成原子化,而解決方案可以使使用redis+lua或者nginx+lua技術(shù)進(jìn)行實(shí)現(xiàn),通過(guò)這兩種技術(shù)可以實(shí)現(xiàn)的高并發(fā)和高性能。
首先我們來(lái)使用redis+lua實(shí)現(xiàn)時(shí)間窗內(nèi)某個(gè)接口的請(qǐng)求數(shù)限流,實(shí)現(xiàn)了該功能后可以改造為限流總并發(fā)/請(qǐng)求數(shù)和限制總資源數(shù)。Lua本身就是一種編程語(yǔ)言,也可以使用它實(shí)現(xiàn)復(fù)雜的令牌桶或漏桶算法。
如下操作因是在一個(gè)lua腳本中(相當(dāng)于原子操作),又因Redis是單線程模型,因此是線程安全的。

相比Redis事務(wù)來(lái)說(shuō),Lua腳本有以下優(yōu)點(diǎn)

減少網(wǎng)絡(luò)開(kāi)銷: 不使用 Lua 的代碼需要向 Redis 發(fā)送多次請(qǐng)求, 而腳本只需一次即可, 減少網(wǎng)絡(luò)傳輸;
原子操作: Redis 將整個(gè)腳本作為一個(gè)原子執(zhí)行, 無(wú)需擔(dān)心并發(fā), 也就無(wú)需事務(wù);
復(fù)用: 腳本會(huì)永久保存 Redis 中, 其他客戶端可繼續(xù)使用.

Lua腳本

local key = KEYS[1] --限流KEY(一秒一個(gè))
local limit = tonumber(ARGV[1]) --限流大小
local current = tonumber(redis.call('get', key) or "0")
if current + 1 > limit then --如果超出限流大小
    return 0
else --請(qǐng)求數(shù)+1,并設(shè)置2秒過(guò)期
    redis.call("INCRBY", key,"1")
    redis.call("expire", key,"2")
end
return 1

java代碼

import org.apache.commons.io.FileUtils;
import redis.clients.jedis.Jedis;
import java.io.File;
import java.io.IOException;
import java.net.URISyntaxException;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.CountDownLatch;
public class RedisLimitRateWithLUA {
    public static void main(String[] args) {
        final CountDownLatch latch = new CountDownLatch(1);
        for (int i = 0; i < 7; i++) {
            new Thread(new Runnable() {
                public void run() {
                    try {
                        latch.await();
                        System.out.println("請(qǐng)求是否被執(zhí)行:"+accquire());
                    } catch (Exception e) {
                        e.printStackTrace();
                    }
                }
            }).start();
        }
        latch.countDown();
    }
    public static boolean accquire() throws IOException, URISyntaxException {
        Jedis jedis = new Jedis("127.0.0.1");
        File luaFile = new File(RedisLimitRateWithLUA.class.getResource("/").toURI().getPath() + "limit.lua");
        String luaScript = FileUtils.readFileToString(luaFile);
        String key = "ip:" + System.currentTimeMillis()/1000; // 當(dāng)前秒
        String limit = "5"; // 最大限制
        List<String> keys = new ArrayList<String>();
        keys.add(key);
        List<String> args = new ArrayList<String>();
        args.add(limit);
        Long result = (Long)(jedis.eval(luaScript, keys, args)); // 執(zhí)行l(wèi)ua腳本,傳入?yún)?shù)
        return result == 1;
    }
}

運(yùn)行結(jié)果

請(qǐng)求是否被執(zhí)行:true
請(qǐng)求是否被執(zhí)行:true
請(qǐng)求是否被執(zhí)行:false
請(qǐng)求是否被執(zhí)行:true
請(qǐng)求是否被執(zhí)行:true
請(qǐng)求是否被執(zhí)行:true
請(qǐng)求是否被執(zhí)行:fals

從結(jié)果可看出只有5個(gè)請(qǐng)求成功執(zhí)行

IP限流Lua腳本

local key = "rate.limit:" .. KEYS[1]
local limit = tonumber(ARGV[1])
local expire_time = ARGV[2]
local is_exists = redis.call("EXISTS", key)
if is_exists == 1 then
    if redis.call("INCR", key) > limit then
        return 0
    else
        return 1
    end
else
    redis.call("SET", key, 1)
    redis.call("EXPIRE", key, expire_time)
    return 1
end

參考 http://www.dbjr.com.cn/books/561366.html

以上就是Redis Lua腳本實(shí)現(xiàn)ip限流示例的詳細(xì)內(nèi)容,更多關(guān)于Redis Lua限流的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Redis憑啥可以這么快

    Redis憑啥可以這么快

    本文詳細(xì)的介紹了為啥使用Redis的時(shí)候,可以做到非??斓淖x取速度,對(duì)于大家學(xué)習(xí)Redis非常有幫助,希望大家喜歡
    2021-02-02
  • redis lua腳本實(shí)戰(zhàn)秒殺和減庫(kù)存的實(shí)現(xiàn)

    redis lua腳本實(shí)戰(zhàn)秒殺和減庫(kù)存的實(shí)現(xiàn)

    本文主要是學(xué)習(xí)一下redis lua腳本的編寫(xiě),以及在redisson這個(gè)redis客戶端中是怎樣使用的,實(shí)戰(zhàn)一下秒殺場(chǎng)景redis減庫(kù)存lua腳本的編寫(xiě),并偽真實(shí)環(huán)境壓測(cè)查看效果。感興趣的可以了解一下
    2021-11-11
  • redis在php中常用的語(yǔ)法【推薦】

    redis在php中常用的語(yǔ)法【推薦】

    string是redis最基本的類型,而且string類型是二進(jìn)制安全的。這篇文章主要介紹了redis在php中常用的語(yǔ)法,需要的朋友可以參考下
    2018-08-08
  • Redis+IDEA實(shí)現(xiàn)單機(jī)鎖和分布式鎖的過(guò)程

    Redis+IDEA實(shí)現(xiàn)單機(jī)鎖和分布式鎖的過(guò)程

    這篇文章主要介紹了Redis+IDEA實(shí)現(xiàn)單機(jī)鎖和分布式鎖的過(guò)程,本文通過(guò)示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2023-07-07
  • Redis中鍵的過(guò)期刪除策略深入講解

    Redis中鍵的過(guò)期刪除策略深入講解

    這篇文章主要給大家介紹了關(guān)于Redis中鍵的過(guò)期刪除策略的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2018-09-09
  • redis基本安裝判斷、啟動(dòng)使用方法示例

    redis基本安裝判斷、啟動(dòng)使用方法示例

    這篇文章主要介紹了redis基本安裝判斷、啟動(dòng)使用方法,結(jié)合實(shí)例形式分析了Redis針對(duì)是否安裝的判斷、啟動(dòng)等使用方法,需要的朋友可以參考下
    2020-02-02
  • redis適合場(chǎng)景八點(diǎn)總結(jié)

    redis適合場(chǎng)景八點(diǎn)總結(jié)

    在本篇文章中我們給大家整理了關(guān)于redis適合什么場(chǎng)景的8點(diǎn)知識(shí)點(diǎn)內(nèi)容,需要的朋友們參考下。
    2019-06-06
  • Redis?哨兵模式的實(shí)現(xiàn)詳解

    Redis?哨兵模式的實(shí)現(xiàn)詳解

    本文主要介紹了Redis?哨兵模式的實(shí)現(xiàn)詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-06-06
  • Redis Template實(shí)現(xiàn)分布式鎖的實(shí)例代碼

    Redis Template實(shí)現(xiàn)分布式鎖的實(shí)例代碼

    使用Redis的SETNX命令獲取分布式鎖的步驟,接下來(lái)通過(guò)本文給大家介紹Redis Template實(shí)現(xiàn)分布式鎖的實(shí)例代碼,代碼簡(jiǎn)單易懂,對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧
    2018-09-09
  • Redis超詳細(xì)講解高可用主從復(fù)制基礎(chǔ)與哨兵模式方案

    Redis超詳細(xì)講解高可用主從復(fù)制基礎(chǔ)與哨兵模式方案

    Redis因?yàn)槠涓咝阅芎鸵子眯栽谖覀兒蠖说姆?wù)中發(fā)揮了巨大的作用,并且很多重要功能的實(shí)現(xiàn)都會(huì)依賴redis,本篇我們來(lái)了解Redis高可用主從復(fù)制與哨兵模式
    2022-04-04

最新評(píng)論