我的编程空间,编程开发者的网络收藏夹
学习永远不晚

React SSR中的限流怎么实现

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

React SSR中的限流怎么实现

这篇文章主要介绍“React SSR中的限流怎么实现”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“React SSR中的限流怎么实现”文章能帮助大家解决问题。

为什么要限流

如下所示是一个简单的 nodejs 服务端项目:

const express = require('express')const app = express()app.get('/', async (req, res) => {  // 模拟 SSR 会大量的占用内存  const buf = Buffer.alloc(1024 * 1024 * 200, 'a')  console.log(buf)  res.end('end')})app.get('/another', async (req, res) => {  res.end('another api')})const listener = app.listen(process.env.PORT || 2048, () => {  console.log('Your app is listening on port ' + listener.address().port)})

其中,我们通过 Buffer 来模拟 SSR 过程会大量的占用内存的情况。

然后,通过 docker build -t ssr . 指定将我们的项目打包成一个镜像,并通过以下命令运行一个容器:

docker run \-it \-m 512m \ # 限制容器的内存--rm \-p 2048:2048 \--name ssr \--oom-kill-disable \ssr

我们将容器内存限制在 512m,并通过 --oom-kill-disable 指定容器内存不足时不关闭容器。

接下来,我们通过 autocannon 来进行一下压测:

autocannon -c 10 -d 1000 http://localhost:2048

通过, docker stats 可以看到容器的运行情况:

CONTAINER ID   NAME      CPU %     MEM USAGE / LIMIT   MEM %     NET I/O           BLOCK I/O         PIDSd9c0189e2b56    ssr     0.00%     512MiB / 512MiB     99.99%    14.6kB / 8.65kB   41.9MB / 2.81MB   40

此时,容器内存已经全部被占用,服务对外失去了响应,通过 curl -m 5 http://localhost:2048 访问,收到了超时的错误提示:

curl: (28) Operation timed out after 5001 milliseconds with 0 bytes received

我们改造一下代码,使用 counter.js 来统计 QPS,并限制为 2:

const express = require('express')const counter = require('./counter.js')const app = express()const limit = 2let cnt = counter()app.get(  '/',  (req, res, next) => {    cnt(1)    if (cnt() > limit) {      res.writeHead(500, {        'content-type': 'text/pain',      })      res.end('exceed limit')      return    }    next()  },  async (req, res) => {    const buf = Buffer.alloc(1024 * 1024 * 200, 'a')    console.log(buf)    res.end('end')  })app.get('/another', async (req, res) => {  res.end('another api')})const listener = app.listen(process.env.PORT || 2048, () => {  console.log('Your app is listening on port ' + listener.address().port)})// counter.jsmodule.exports = function counter(interval = 1000) {  let arr = []  return function cnt(number) {    const now = Date.now()    if (number > 0) {      arr.push({        time: now,        value: number,      })      const newArr = []      // 删除超出一秒的数据      for (let i = 0, len = arr.length; i < len; i++) {        if (now - arr[i].time > interval) continue        newArr.push(arr[i])      }      arr = newArr      return    }    // 计算前一秒的数据和    let sum = 0    for (let i = arr.length - 1; i >= 0; i--) {      const {time, value} = arr[i]      if (now - time <= interval) {        sum += value        continue      }      break    }    return sum  }}

此时,容器运行正常:

CONTAINER ID   NAME      CPU %     MEM USAGE / LIMIT   MEM %     NET I/O           BLOCK I/O        PIDS3bd5aa07a3a7   ssr     88.29%    203.1MiB / 512MiB   39.67%    24.5MB / 48.6MB   122MB / 2.81MB   40

虽然此时访问 / 路由会收到错误:

curl -m 5  http://localhost:2048exceed limit

但是 /another 却不受影响:

curl -m 5  http://localhost:2048/anotheranother api

由此可见,限流确实是系统进行自我保护的一个比较好的方法。

令牌桶算法

常见的限流算法有“滑动窗口算法”、“令牌桶算法”,我们这里讨论 “令牌桶算法” 。在令牌桶算法中,存在一个桶,容量为 burst 。该算法以一定的速率(设为 rate )往桶中放入令牌,超过桶容量会丢弃。每次请求需要先获取到桶中的令牌才能继续执行,否则拒绝。

根据令牌桶的定义,我们实现令牌桶算法如下:

export default class TokenBucket {  private burst: number  private rate: number  private lastFilled: number  private tokens: number  constructor(burst: number, rate: number) {    this.burst = burst    this.rate = rate    this.lastFilled = Date.now()    this.tokens = burst  }  setBurst(burst: number) {    this.burst = burst    return this  }  setRate(rate: number) {    this.rate = rate    return this  }  take() {    this.refill()    if (this.tokens > 0) {      this.tokens -= 1      return true    }    return false  }  refill() {    const now = Date.now()    const elapse = now - this.lastFilled    this.tokens = Math.min(this.burst, this.tokens + elapse * (this.rate / 1000))    this.lastFilled = now  }}

然后,按照如下方式使用:

const tokenBucket = new TokenBucket(5, 10)if (tokenBucket.take()) {  // Do something} else {  // refuse}

简单解释一下这个算法,调用 take 时,会先执行 refill 先往桶中进行填充。填充的方式也很简单,首先计算出与上次填充的时间间隔 elapse 毫秒,然后计算出这段时间内应该补充的令牌数,因为令牌补充速率是 rate 个/秒,所以需要补充的令牌数为:

elapse * (this.rate / 1000)

又因为令牌数不能超过桶的容量,所以补充后桶中的令牌数为:

Math.min(this.burst, this.tokens + elapse * (this.rate / 1000))

注意,这个令牌数是可以为小数的。

令牌桶算法具有以下两个特点:

  • 当外部请求的 QPS M 大于令牌补充的速率 rate 时,长期来看,最终有效的 QPS 会趋向于 rate 。这个很好理解,拉的总不可能比吃的多吧。

  • 因为令牌桶可以存下 burst 个令牌,所以可以允许短时间的激增流量,持续的时间为:

T = burst / (M - rate) // rate < M

可以理解为一个水池里面有 burst 的水量,进水的速率为 rate ,出水的速率为 M ,则净出水速率为 M-rate ,则水池中的水放空的时间即为激增流量的持续时间。

关于“React SSR中的限流怎么实现”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识,可以关注编程网行业资讯频道,小编每天都会为大家更新不同的知识点。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

React SSR中的限流怎么实现

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

React SSR中的限流怎么实现

这篇文章主要介绍“React SSR中的限流怎么实现”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“React SSR中的限流怎么实现”文章能帮助大家解决问题。为什么要限流如下所示是一个简单的 nod
2023-07-02

ASP.NET Core中间件怎么实现限流

本篇内容介绍了“ASP.NET Core中间件怎么实现限流”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!一、限流算法在高并发系统中,有三把利
2023-06-29

Python怎么实现简易的限流器

今天小编给大家分享一下Python怎么实现简易的限流器的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。简单总结就是:动态的re
2023-06-29

kubernetes怎么实现分布式限流

本篇内容主要讲解“kubernetes怎么实现分布式限流”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“kubernetes怎么实现分布式限流”吧!一、概念限流(Ratelimiting)指对应用
2023-06-29

Java常见的限流算法怎么实现

这篇“Java常见的限流算法怎么实现”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Java常见的限流算法怎么实现”文章吧。为
2023-06-29

java限流功能redis怎么实现

这篇指南介绍了如何使用Redis实现Java应用程序的限流功能。它详细介绍了令牌桶、漏桶和滑动窗口算法的Redis实现,并提供了选择合适算法和实施其他注意事项的指导。
java限流功能redis怎么实现
2024-04-02

Redis怎么实现限流和熔断

Redis 可以通过以下方式实现限流和熔断:限流:使用 Redis 的计数器功能来实现限流。在每次请求到来时,首先检查计数器的值是否超过设定的阈值,如果超过则拒绝请求,否则允许请求并将计数器加一。可以使用 Redis 的 INCR 命令来实
Redis怎么实现限流和熔断
2024-05-07

PHP怎么实现令牌桶限流

小编给大家分享一下PHP怎么实现令牌桶限流,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!PHP实现令牌桶限流的方法:1、设有一个令牌桶,桶内存放令牌;2、每次访问
2023-06-25

Golang怎么实现常见的限流算法

本篇内容介绍了“Golang怎么实现常见的限流算法”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!固定窗口每开启一个新的窗口,在窗口时间大小内
2023-07-05

React中怎么实现插槽

今天小编给大家分享一下React中怎么实现插槽的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。React中实现插槽设计插槽在R
2023-07-05

java怎么使用Semaphore实现限流器

这篇文章主要讲解了“java怎么使用Semaphore实现限流器”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“java怎么使用Semaphore实现限流器”吧!概念1、Semaphore可以
2023-06-30

react中的DOM操作怎么实现

这篇“react中的DOM操作怎么实现”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“react中的DOM操作怎么实现”文章吧
2023-06-05

怎么在java中利用Semaphore实现一个限流器

怎么在java中利用Semaphore实现一个限流器?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。Java可以用来干什么Java主要应用于:1. web开发;2
2023-06-14

React中的合成事件怎么实现

这篇文章主要介绍了React中的合成事件怎么实现的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇React中的合成事件怎么实现文章都会有所收获,下面我们一起来看看吧。1 事件三个阶段 捕获、目标、处理 (具体百度
2023-07-05

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录