百度网盘不限速被限速_基本API限速

百度网盘不限速被限速

您可能正在开发某种形式的(Web / RESTful)API,并且如果它是面向公众的(甚至是内部的),则通常需要以某种方式对其进行速率限制。 即,限制一段时间内执行的请求数,以节省资源并防止滥用。

这可能可以通过一些聪明的配置在Web服务器/负载均衡器级别上实现,但是通常您希望速率限制器是特定于客户端的(即,API的每个客户端都应具有单独的速率限制),以及客户端的方式被确定是不同的。 可能仍然可以在负载均衡器上执行此操作,但是我认为将其放在应用程序级别上是有意义的。

我将使用spring-mvc作为示例,但是任何Web框架都有插入拦截器的好方法。

所以这是一个spring-mvc拦截器的例子:

@Component
public class RateLimitingInterceptor extends HandlerInterceptorAdapter {private static final Logger logger = LoggerFactory.getLogger(RateLimitingInterceptor.class);@Value("${rate.limit.enabled}")private boolean enabled;@Value("${rate.limit.hourly.limit}")private int hourlyLimit;private Map<String, Optional<SimpleRateLimiter>> limiters = new ConcurrentHashMap<>();@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)throws Exception {if (!enabled) {return true;}String clientId = request.getHeader("Client-Id");// let non-API requests passif (clientId == null) {return true;}SimpleRateLimiter rateLimiter = getRateLimiter(clientId);boolean allowRequest = limiter.tryAcquire();if (!allowRequest) {response.setStatus(HttpStatus.TOO_MANY_REQUESTS.value());}response.addHeader("X-RateLimit-Limit", String.valueOf(hourlyLimit));return allowRequest;}private SimpleRateLimiter getRateLimiter(String clientId) {if (limiters.containsKey(clientId)) {return limiters.get(clientId);} else {synchronized(clientId.intern()) {// double-checked locking to avoid multiple-reinitializationsif (limiters.containsKey(clientId)) {return limiters.get(clientId);}SimpleRateLimiter rateLimiter = createRateLimiter(clientId);limiters.put(clientId, rateLimiter);return rateLimiter;}}}@PreDestroypublic void destroy() {// loop and finalize all limiters}
}

这将按需初始化每个客户端的速率限制器。 另外,在启动时,您可以遍历所有已注册的API客户端,并为每个客户端创建一个速率限制器。 如果速率限制器不允许更多请求(tryAcquire()返回false),则取消“ Too many requests”(太多请求)并中止请求的执行(从拦截器返回“ false”)。

这听起来很简单。 但是有一些问题。 您可能想知道上面的SimpleRateLimiter在哪里定义。 我们将到达那里,但首先让我们看看我们对速率限制器实现有哪些选择。

最受欢迎的似乎是番石榴RateLimiter 。 它具有简单的工厂方法,可为您提供指定速率(每秒允许)的速率限制器。 但是,它不能很好地适应Web API,因为您无法使用预先存在的许可数量初始化RateLimiter。 这意味着在限制器允许请求之前,应经过一段时间。 还有另一个问题–如果您每秒的许可数量少于一个(例如,如果您希望的速率限制为“每小时200个请求”),则可以传递一个分数(hourlyLimit / secondsInHour),但仍然无法达到您的目的可以预期,因为内部有一个“ maxPermits”字段,可以将许可证数量的上限限制为比您想要的要少得多。 此外,速率限制器不允许突发-您每秒恰好有X个许可,但您不能长时间分散它们,例如在一秒钟内有5个请求,然后在接下来的几秒钟内没有请求。 实际上,上述所有问题都可以解决,但遗憾的是,可以通过您无法访问的隐藏字段来解决。 多年来,存在多个功能请求,但是Guava不会更新速率限制器,从而使其不适用于API速率限制。

使用反射,您可以调整参数并使限制器工作。 但是,这很丑陋,并且不能保证它会按预期工作。 我在这里展示了如何使用每小时X许可,爆破性和完整的初始许可来初始化番石榴速率限制器。 当我认为这样做的时候,我看到tryAcquire()有一块tryAcquire() synchronized(..)块。 这是否意味着在仅检查是否允许发出请求时,所有请求都会彼此等待? 那太可怕了。

因此,实际上番石榴RateLimiter并非旨在(网络)API速率限制。 番石榴(Guava)劝阻人们不要滥用它的方法,也许保持其功能欠佳?

这就是为什么我决定根据Java信号量自己实现一些简单的事情。 这是朴素的实现 :

public class SimpleRateLimiter {private Semaphore semaphore;private int maxPermits;private TimeUnit timePeriod;private ScheduledExecutorService scheduler;public static SimpleRateLimiter create(int permits, TimeUnit timePeriod) {SimpleRateLimiter limiter = new SimpleRateLimiter(permits, timePeriod);limiter.schedulePermitReplenishment();return limiter;}private SimpleRateLimiter(int permits, TimeUnit timePeriod) {this.semaphore = new Semaphore(permits);this.maxPermits = permits;this.timePeriod = timePeriod;}public boolean tryAcquire() {return semaphore.tryAcquire();}public void stop() {scheduler.shutdownNow();}public void schedulePermitReplenishment() {scheduler = Executors.newScheduledThreadPool(1);scheduler.schedule(() -> {semaphore.release(maxPermits - semaphore.availablePermits());}, 1, timePeriod);}
}

它需要一定数量的许可(允许的请求数量)和一段时间。 时间段为“ 1 X”,其中X可以是秒/分钟/小时/每天-取决于您希望如何配置限制-每秒,每分钟,每小时,每天。 调度程序每1 X补充所获得的许可证。 无法控制突发事件(客户端可以用快速连续的请求花费所有许可),没有热身功能,没有逐步的补充。 根据您的需要,这可能并不理想,但这只是一个基本的速率限制器,它是线程安全的,没有任何阻塞。 我编写了一个单元测试,以确认限制器的行为正确,并且还对本地应用程序进行了性能测试,以确保遵守限制。 到目前为止,它似乎正在工作。

有其他选择吗? 好吧,是的–像RateLimitJ这样的库使用Redis来实现速率限制。 但是,这意味着您需要设置和运行Redis。 对于“简单地”进行速率限制,这似乎是开销。

另一方面,限速将如何在一组应用程序节点中正常工作? 应用程序节点可能需要一些数据库或八卦协议来共享有关剩余的每个客户端许可(请求)的数据? 不必要。 解决此问题的一种非常简单的方法是假设负载均衡器在节点之间平均分配负载。 这样,您只需要将每个节点上的限制设置为等于总限制除以节点数即可。 它不是很精确,但是您很少需要做到这一点–允许5-10个以上的请求不会终止您的应用程序,允许5-10个以下的请求对用户来说并不算太大。

但是,那将意味着您必须知道应用程序节点的数量。 如果您使用自动缩放(例如在AWS中),则节点数可能会根据负载而变化。 在这种情况下,补给排定的作业无需配置硬编码的许可证数量,而是可以通过调用AWS(或其他云提供商)API来获取其中的节点数量,从而动态计算“ maxPermits”。当前的自动缩放组。 这比仅支持Redis部署要简单得多。

总的来说,我很惊讶没有一种“规范”的方法来实现速率限制(在Java中)。 限制速率的需求可能并不像看起来那样普遍。 或者,它是手动实施的–通过临时禁止使用“资源过多”的API客户端。

翻译自: https://www.javacodegeeks.com/2017/07/basic-api-rate-limiting.html

百度网盘不限速被限速

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/335071.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

mysql怎么查看刷脏页慢_一条SQL查询语句极为缓慢,如何去优化呢

一条 SQL 查询语句执行的很慢&#xff0c;那是每次查询都很慢呢&#xff1f;还是大多数情况下是正常的&#xff0c;偶尔出现很慢呢&#xff1f;可以分以下两种情况来讨论。大多数情况是正常的&#xff0c;只是偶尔会出现很慢的情况。在数据量不变的情况下&#xff0c;这条SQL语…

MacBook Air如何清理缓存

方法一&#xff1a;使用热键 AltCmdRP电源键这是清理缓存热键。 用法&#xff1a;先按住前4个键&#xff0c;然后按一次电源键&#xff0c;前四个键不松开&#xff0c;等到听到电脑第四次开机音效后可以松开&#xff0c;开机音效共有五次&#xff0c;然后电脑自动开机 方法二&…

apache ignite_使用Spring Data的Apache Ignite

apache igniteSpring Data提供了一种统一而简便的方法来访问不同类型的持久性存储&#xff0c;关系数据库系统和NoSQL数据存储。 它位于JPA之上&#xff0c;添加了另一层抽象并定义了基于标准的设计以在Spring上下文中支持持久层。 Apache Ignite IgniteRepository实现了Spri…

centos 7 mysql随机密码_在centos中安装了mysql5.7之后解决不知道随机的密码的问题...

在安装完成mysql5.7 之后&#xff0c;发现密码不知道。不要紧&#xff0c;直接重置密码。1.修改配置文件my.cfg[rootlocalhost ~]# vi /etc/my.cnf找到mysqld在之后添加skip-grant-tables保存退出2. 重启mysql服务service mysqld restart3.直接登陆mysql而不需要密码mysql -u…

python单用户登录_Django实现单用户登录的方法示例

最近由于要毕业了写论文做毕设&#xff0c;然后还在实习发现已经好久都没有写博客了。今天由于工作需求&#xff0c;需要用Django实现单用户登录。大概意思就是跟QQ一样的效果&#xff0c;每个账号只能一个地方登录使用&#xff0c;限制账号的登录次数。由于用的是Django自带的…

IntelliJ IDEA for Windows 默认模式下的快捷键

文章目录General 通用Debugging 调试Search/ Replace 查询/替换Editing 编辑Refactoring 重构Navigation 导航Compile and Run 编译和运行Usage Search 使用查询VCS/ Local History 版本控制/本地历史记录Live Templates 动态代码模板Other 官方文档上没有体现点击查看官方文档…

java事件处理过程分布写_Java 9中的进程处理

java事件处理过程分布写一直以来&#xff0c;用Java管理操作系统进程都是一项艰巨的任务。 这样做的原因是可用的工具和API较差。 老实说&#xff0c;这并非没有道理&#xff1a;Java并非旨在达到目的。 如果要管理OS进程&#xff0c;则可以使用所需的Shell&#xff0c;Perl脚本…

mac mysql5.7.9 dmg_Mac 安装 mysql5.7

mac 安装msql 5.7最近使用Mac系统&#xff0c;准备搭建一套本地web服务器环境。因为Mac系统没有自带mysql&#xff0c;所以要手动去安装mysql&#xff0c;本次安装mysql最新版5.7.28。安装步骤参考以下博客https://www.jianshu.com/p/71f81a0c62b2安装成功后&#xff0c;因为密…

安卓系统dicom阅读器_用户分享:电子书阅读器Note Pro,一座贴心的移动图书馆...

本文转载自“什么值得买”官网用户“小良读书”&#xff0c;经作者授权转载。Note Pro&#xff0c;一座贴心的移动图书馆移动图书馆貌美的小书郎10.3寸高清大屏更适合专业书籍的阅读如果说多年前入手了一台kindle paperwite3电纸书阅读器&#xff0c;它让我畅游了书籍的江河&am…

vim 编辑器命令整理

文章目录一、基本使用流程二、普通命令模式&#xff08;一&#xff09;切换到插入模式&#xff08;编辑/写入/输入&#xff09;&#xff08;二&#xff09;切换到可视模式&#xff08;选择文本模式&#xff09;&#xff08;三&#xff09;切换至底行命令模式&#xff08;四&…

mysql f_MySQL

医生给病人诊断的时候&#xff0c;一般会使用听诊器来诊断肺部是否正常。如果你的MySQL出现了性能问题&#xff0c;第一个需要“诊断”的就是slow log(慢日志)了。slow log文件很小&#xff0c;使用more less等命令就足够了。如果slow log很大怎么办&#xff1f;这里介绍MySQL自…

activiti dmn_端到端BPM(带有DMN标记)

activiti dmn下周的红帽峰会即将成为有史以来最好的峰会之一&#xff01; 而且&#xff0c;如果您是Drools或jBPM的狂热者&#xff0c;您会很忙 &#xff1a;Signavio和Red Hat之间的合作伙伴关系是我们为您准备的另一个顶级演讲。 邓肯道尔&#xff08;Duncan Doyle&#xff…

python计算两个点之间的距离_python实现两个经纬度点之间的距离和方位角的方法...

最近做有关GPS轨迹上有关的东西&#xff0c;花费心思较多&#xff0c;对两个常用的函数总结一下&#xff0c;求距离和求方位角&#xff0c;比较精确&#xff0c;欢迎交流&#xff01;1. 求两个经纬点的方位角&#xff0c;P0(latA, lonA), P1(latB, lonB)(很多博客写的不是很好&…

netbeans 9_NetBeans 9抢先体验

netbeans 9Java 9即将来临&#xff0c;NetBeans 9也即将来临。在本文中&#xff0c;我们将看到NetBeans 9 Early Access为开发人员提供的支持&#xff0c;以帮助他们构建Java 9兼容的应用程序。 Java 9提供了许多&#xff08;大约90种&#xff09; 新功能&#xff0c;包括模块…

同一个ip能否两次加入组播_组播IGMPv1/v2/v3精华知识汇总

闲言少叙&#xff0c;直奔主题&#xff0c;开说之前先做点铺垫&#xff0c;回顾一下网络的基础知识。D类地址划分回顾组播是IPv4地址分类中的D类&#xff0c;我们回顾一下IPv4的地址划分A类&#xff1a;首bit0&#xff0c;0-127网段&#xff0c;网络掩码8&#xff0c;其中0.0.0…

命令(CMD)终端的清屏(清除/清空)命令/快捷键

Windows CMD 或者 Navicat 命令窗口的清屏命令&#xff1a;Clear 或者 CLS MacBook 命令终端的清屏命令/快捷键&#xff1a;CtrlCmdL&#xff08;清除屏幕&#xff09; &#xff0c;CmdK&#xff08;清除到开头&#xff09;&#xff0c;Clear &#xff0c;CtrlL Linux 命令终…

弹窗要打开或保存来自_如何让 PopClip 支持印象笔记客户端:保存到印象笔记amp;高亮文字...

自从印象笔记品牌独立并发布新APP后&#xff0c;Mac系统下很多工具不再支持与印象笔记客户端进行联动&#xff0c;PopClip 就是其中之一。目前 Popclip 官方并未将印象笔记加入到插件的支持中&#xff0c;不过我们可以手动修改来解决这个问题。Popclip 之所以不支持新版本的印象…

vue 混入对象_特性和混入不是面向对象的

vue 混入对象让我立刻说&#xff0c;我们将在这里讨论的功能是那些迫切需要进行放线手术的人带给面向对象编程的纯粹的毒药 &#xff0c;就像David West在他的《 Object Thinking》一书中所建议的那样。 这些功能具有不同的名称&#xff0c;但是最常见的是traits和mixins 。 我…

glup node 内存不够_Redis:内存被我用完了!该怎么办?

介绍Redis是一个内存数据库&#xff0c;当Redis使用的内存超过物理内存的限制后&#xff0c;内存数据会和磁盘产生频繁的交换&#xff0c;交换会导致Redis性能急剧下降。所以在生产环境中我们通过配置参数maxmemoey来限制使用的内存大小。当实际使用的内存超过maxmemoey后&…

Windows应用程序启动命令汇总

winver---------检查Windows版本 wmimgmt.msc----打开windows管理体系结构(WMI) wupdmgr--------windows更新程序 wscript--------windows脚本宿主设置 write----------写字板 winmsd---------系统信息 wiaacmgr-------扫描仪和照相机向导 winchat--------XP自带局域网聊天 me…