成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

神器!API接口限流就是這么簡(jiǎn)單

開(kāi)發(fā) 前端
訪問(wèn)速率限制是一種API訪問(wèn)限制的策略。它限制客戶端在一定時(shí)間內(nèi)調(diào)用 API 的次數(shù)。這有助于保護(hù)應(yīng)用程序接口,防止無(wú)意或惡意的過(guò)度使用。

環(huán)境:SpringBoot3.2.5

1. 簡(jiǎn)介

訪問(wèn)速率限制是一種API訪問(wèn)限制的策略。它限制客戶端在一定時(shí)間內(nèi)調(diào)用 API 的次數(shù)。這有助于保護(hù)應(yīng)用程序接口,防止無(wú)意或惡意的過(guò)度使用。

速率限制通常是通過(guò)跟蹤 IP 地址或更具體的業(yè)務(wù)方式(如 API 密鑰或訪問(wèn)令牌等方式)來(lái)應(yīng)用于 API 的。作為 API 開(kāi)發(fā)人員,當(dāng)客戶端達(dá)到限制時(shí),我們有幾種選擇:

  • 請(qǐng)求排隊(duì),直到剩余時(shí)間結(jié)束(這也是最常用的方式)
  • 拒絕請(qǐng)求(HTTP 429 請(qǐng)求過(guò)多)

本篇文章將介紹一款開(kāi)源的組件Bucket4j,該組件提供了強(qiáng)大的限流功能?;诨诹钆仆八惴ā<瓤捎糜讵?dú)立的 JVM 應(yīng)用程序,也可用于集群環(huán)境。它還通過(guò) JCache(JSR107)規(guī)范支持內(nèi)存或分布式緩存。

令牌桶算法

假設(shè)我們有一個(gè) "桶",其容量被定義為可容納的令牌數(shù)量。每當(dāng)消費(fèi)者想要訪問(wèn) API 端點(diǎn)時(shí),就必須從桶中獲取一個(gè)令牌。如果有令牌,我們就會(huì)從數(shù)據(jù)桶中移除令牌,并接受請(qǐng)求。反之,如果程序桶中沒(méi)有令牌,我們就會(huì)拒絕請(qǐng)求。

在請(qǐng)求消耗令牌(token)的同時(shí),我們也在以某種固定的速度補(bǔ)充令牌。

考慮一個(gè)速率限制為每分鐘 100 個(gè)請(qǐng)求的應(yīng)用程序接口。我們可以創(chuàng)建一個(gè)容量為 100 的水桶,每分鐘填充 100 個(gè)令牌。如果我們?cè)谝环昼妰?nèi)收到 70 個(gè)請(qǐng)求,少于可用令牌的數(shù)量,那么在下一分鐘開(kāi)始時(shí),我們只需再添加 30 個(gè)令牌,就能使水桶達(dá)到容量。另一方面,如果我們?cè)?40 秒內(nèi)用完了所有令牌,我們將等待 20 秒來(lái)重新裝滿令牌桶。

接下來(lái)將詳細(xì)介紹在Spring Boot中如何使用Bucket4j實(shí)現(xiàn)限流。

2. 實(shí)戰(zhàn)案例

2.1 環(huán)境準(zhǔn)備

引入依賴

<dependency>
  <groupId>com.giffing.bucket4j.spring.boot.starter</groupId>
  <artifactId>bucket4j-spring-boot-starter</artifactId>
  <version>0.12.7</version>
</dependency>
<dependency>
  <groupId>com.bucket4j</groupId>
  <artifactId>bucket4j-redis</artifactId>
  <version>8.10.1</version>
</dependency>
<dependency>
  <groupId>redis.clients</groupId>
  <artifactId>jedis</artifactId>
</dependency>
<dependency>
  <groupId>io.micrometer</groupId>
  <artifactId>micrometer-core</artifactId>
</dependency>

接下來(lái)的案例是基于redis的,所以引入了jedis。你也可以是lettuce或者是redisson但是這2個(gè)貌似需要是webflux環(huán)境。

jedis配置

@Bean
public JedisPool jedisPool(
    @Value("${spring.data.redis.port}") Integer port,
    @Value("${spring.data.redis.host}") String host,
    @Value("${spring.data.redis.password}") String password,
    @Value("${spring.data.redis.database}") Integer database
  ) {
  // buildPoolConfig方法自己進(jìn)行配置吧
  final JedisPoolConfig poolConfig = buildPoolConfig();
  return new JedisPool(poolConfig, host, port, 60000, password, database);
}

以上基礎(chǔ)環(huán)境就準(zhǔn)備好了,接下來(lái)就可以進(jìn)行規(guī)則配置。而規(guī)則的配置可以基于2中方式,基于配置文件和基于注解(AOP)。

定義接口

@RestController
@RequestMapping("/products")
public class ProductController {


  @GetMapping("/{id}")
  public Product getProduct(@PathVariable Integer id) {
    return new Product(id, "商品 - " + id, BigDecimal.valueOf(new Random().nextDouble(1000))) ;
  }
}

接下來(lái)我將基于上面的接口進(jìn)行限流的配置。

2.2 基于配置文件

基于配置文件的規(guī)則配置底層實(shí)現(xiàn)是通過(guò)Filter。

bucket4j:
  cache-to-use: redis-jedis
  filter-config-caching-enabled: true
  filters:
  - cache-name: product_cache_name
    id: product_filter
    # 配置請(qǐng)求url的規(guī)則;這里底層是通過(guò)正則表達(dá)式進(jìn)行匹配的
    url: /products/.*
    rate-limits:
    - 
      #這里的cache-key非常關(guān)鍵;用于區(qū)分不同請(qǐng)求的情況;
      #比如,這里我會(huì)根據(jù)不同的請(qǐng)求id來(lái)現(xiàn)在訪問(wèn)速率
      #這里可以寫spel表達(dá)式,這里調(diào)用的是HttpServletRequest#getParameter方法
      cache-key: getParameter("id")
      bandwidths:
      #配置桶的容量
      - capacity: 2
        # 時(shí)間
        time: 30
        # 單位
        unit: seconds
        # 填充速度;這會(huì)每隔30秒進(jìn)行填充
        refill-speed: interval

如何配置后,訪問(wèn)/products/xxx接口,結(jié)果如下:

圖片圖片

圖片圖片

分別請(qǐng)求id為1,2時(shí),如果超過(guò)了配置桶的容量,則默認(rèn)顯示上面的錯(cuò)誤消息,同時(shí)Redis中會(huì)分別記錄這2個(gè)key

圖片圖片

只有等到30s后才能繼續(xù)訪問(wèn)。

修改默認(rèn)的限流提示

bucket4j:
  filters:
  - cache-name: product_cache_name
    http-content-type: 'application/json;charset=utf-8'
    http-response-body: '{"code": -1, "message": "請(qǐng)求太快了"}'

注意:你必須同時(shí)要設(shè)置content-type設(shè)置字符編碼,否則會(huì)亂碼。

條件放行

你也可以通過(guò)如下屬性進(jìn)行有條件的放行;

bucket4j:
  filters:
  - cache-name: product_cache_name
    rate-limits:
    - 
      skip-condition: 'getParameter("id").equals("6")'

當(dāng)請(qǐng)求id的值為6時(shí)則跳過(guò)規(guī)則,直接方向。

以上是基于配置文件規(guī)則的應(yīng)用,它還有很多其它的配置屬性,詳細(xì)查看官方文檔

https://github.com/MarcGiffing/bucket4j-spring-boot-starter

接下來(lái)介紹基于注解的方式。

2.3 基于注解

通過(guò)"@RateLimiting"注解,AOP 可以攔截目標(biāo)方法。這樣,你就可以全面訪問(wèn)方法參數(shù),輕松定義速率限制鍵或有條件地跳過(guò)速率限制。

配置文件中配置規(guī)則

bucket4j:
  methods:
  - name: storage_rate #在代碼中會(huì)通過(guò)該名稱引用
    cache-name: storage_cache_name
    rate-limit:
      bandwidths:
      - capacity: 2
        time: 30
        unit: seconds
        refill-speed: interval

接口注解,配置限流

@RateLimiting(
    name = "storage_rate", 
    cacheKey = "'storage-' + #id",
    skipCondition = "#name eq 'admin'",
    ratePerMethod = true,
    fallbackMethodName = "getStorageFallback"
  )
@GetMapping("/{id}")
public R<Storage> getStorage(@PathVariable Integer id, String name) {
  return R.success(new Storage(id, "SP001 - " + id, new Random().nextInt(10000))) ;
}
// 回退方法的簽名必須與業(yè)務(wù)方法一致
public R<Storage> getStorageFallback(Integer id, String name) {
  return R.failure(String.format("請(qǐng)求id=%d,name=%s被限流", id, name)) ;
}

skipCondition:該屬性定義了如果請(qǐng)求的name的值為admin則跳過(guò),不限流。

圖片圖片

圖片圖片

@RateLimiting注解還可以應(yīng)用到類中,這樣該類中的所有方法都會(huì)被限流,如下示例:

@Service
@RateLimiting(
    name = "storage_rate", 
    cacheKey = "getName",
    ratePerMethod = false
  )
public class StorageService {


  public Storage queryStorageById(Integer id) {
    return new Storage(id, "SP001 - " + id, new Random().nextInt(10000)) ;
  }
  
  @IgnoreRateLimiting
  public List<Storage> queryStorages() {
    return List.of(
        new Storage(1, "SP001 - " + 1, new Random().nextInt(10000)),
        new Storage(2, "SP002 - " + 2, new Random().nextInt(10000)),
        new Storage(3, "SP003 - " + 3, new Random().nextInt(10000))
      ) ;
  }
}

上面代碼queryStorageById會(huì)被限流,而queryStorages方法被@IgnoreRateLimiting注解標(biāo)準(zhǔn),所以不會(huì)被限流。

關(guān)于Bucket4j就介紹這么多,它還有非常多的其它功能,你可以參考官方文檔進(jìn)行查看。

責(zé)任編輯:武曉燕 來(lái)源: Spring全家桶實(shí)戰(zhàn)案例源碼
相關(guān)推薦

2024-09-09 11:35:35

2024-12-03 08:43:49

2023-08-21 08:01:03

2017-11-28 15:29:04

iPhone X網(wǎng)頁(yè)適配

2021-05-24 10:50:10

Git命令Linux

2020-06-16 10:57:20

搭建

2016-07-22 15:12:12

Win10技巧重裝

2021-02-26 10:21:35

比特幣投資金融

2019-05-27 14:03:48

開(kāi)發(fā)技能代碼

2020-04-20 10:47:57

Redis數(shù)據(jù)開(kāi)發(fā)

2023-08-26 21:42:08

零拷貝I/O操作

2021-12-27 07:31:37

JavaNeo4J數(shù)據(jù)庫(kù)

2023-07-27 08:26:36

零拷貝I/O操作

2019-05-13 08:24:58

數(shù)據(jù)庫(kù)MySQLInnoDB

2021-10-28 19:23:27

界面嵌入式 UI

2024-02-27 08:14:51

Nginx跨域服務(wù)

2013-06-09 10:34:24

華為網(wǎng)絡(luò)規(guī)劃企業(yè)ICT

2020-07-27 07:00:00

超文本鏈接Word文檔網(wǎng)絡(luò)

2020-06-17 11:42:50

異常解析器Spring MVC

2017-11-06 16:30:33

開(kāi)源
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

主站蜘蛛池模板: 欧美午夜在线 | 国产乱码精品一区二区三区五月婷 | 精品一区在线免费观看 | 国产福利资源在线 | 国产精品揄拍一区二区 | 日韩精品 | 黄网站涩免费蜜桃网站 | 毛片韩国 | 自拍视频网 | 午夜在线视频一区二区三区 | 国产高清精品一区二区三区 | 亚洲精品久久久久久久久久久久久 | 国内久久精品 | 日韩午夜在线播放 | 免费h在线 | 伊人色综合久久久天天蜜桃 | 国产精品成av人在线视午夜片 | 天天天天天天操 | 精品99爱视频在线观看 | 一区二区免费视频 | 国产成人免费视频 | 久久精品美女 | 毛片av免费在线观看 | 国产网站在线免费观看 | 日韩伦理一区二区 | 亚洲九九精品 | 欧美亚洲高清 | 91久色| 日本一区二区三区四区 | 国产91av视频 | 欧美日韩亚洲视频 | 999久久久精品 | 浮生影院免费观看中文版 | 美女福利视频网站 | 毛片a| 新91| jdav视频在线观看免费 | 成人影院午夜 | 欧美 日韩 国产 成人 在线 | 一区二区三区免费观看 | 91视视频在线观看入口直接观看 |