百度网盘不限速被限速

您可能正在开发某种形式的(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

百度网盘不限速被限速

百度网盘不限速被限速_基本API限速相关推荐

  1. 百度网盘偷偷更新,终于实现免费不限速了

    给你讲个笑话:百度网盘的下载速度. 作为国内仅剩的几款网盘之一,它的占有率是绝对的No.1.但一直以来,百度网盘限速都是一个老生常谈的问题. 动辄几十KB/s,甚至几KB/s,慢的让人痛不欲生.普通用 ...

  2. soap 版本可能不匹配: 出现意外的 envelope 命名空间_10M/S!百度网盘偷偷更新,终于实现免费不限速了!...

    给你讲个笑话:百度网盘的下载速度. 作为国内仅剩的几款网盘之一,它的占有率是绝对的No.1.但一直以来,百度网盘限速都是一个老生常谈的问题. 动辄几十KB/s,甚至几KB/s,慢的让人痛不欲生.普通用 ...

  3. 10M/S!百度网盘偷偷更新,终于实现免费不限速了!

    来自:扩展迷EXTFANS(ID:infinitydaily).快科技 阅读文本大概需要 5 分钟 给你讲个笑话:百度网盘的下载速度. 作为国内仅剩的几款网盘之一,它的占有率是绝对的No.1.但一直以 ...

  4. 免费空间python_10M/S!百度网盘偷偷更新,终于实现免费不限速了?!

    给你讲个笑话:百度网盘的下载速度. 作为国内仅剩的几款网盘之一,它的占有率是绝对的No.1.但一直以来,百度网盘限速都是一个老生常谈的问题. 动辄几十KB/s,甚至几KB/s,慢的让人痛不欲生.普通用 ...

  5. 百度网盘分享链接的提取码查询API

    1.API地址 https://nuexini.gq/bdp.php?url= 2.使用方法 url=后面跟百度网盘分享链接,例如我的网盘链接为:https://pan.baidu.com/s/1FO ...

  6. aria2百度网盘 http返回头无效_接口测试第10期:Fiddler中查看HTTP请求中的状态码...

    关注后回复[软件测试]查看更多上期介绍了常见的HTTP请求方法,get请求.post请求,今天介绍HTTP请求中的状态码,以后也能分辨出哪些状态码是表示客户端错误,哪些是服务器错误.状态码在Fiddl ...

  7. 官方整顿网盘限速,网友齐刷刷@百度网盘

    划重点: 1.作为网盘市场的老大哥,百度网盘拥有7亿用户,占据85%的市场份额,但过度限速也使其成为众矢之的.有媒体评论道,吃相过于难看. 2."基本的下载需求"是一个相对模糊的概 ...

  8. 百度网盘加速无限试用_单次付费来了!百度网盘推出3元/5分钟加速下载服务

    在几天前的7月24日召开的百度网盘媒体沟通会上,百度网盘总经理汤利华表示,百度网盘将改变单一式的会员套餐付费模式,推出更加灵活和精细化的付费模式. 而现在,百度网盘已推出了单次时长的会员收费加速下载服 ...

  9. 再来一个吊打百度网盘的开源神器,还是99年妹子开发的

    点击上方"视学算法",星标公众号 重磅干货,第一时间送达 来源丨开源最前线(ID:OpenSourceTop) 猿妹整编 百度网盘大家都很熟悉,大家都知道百度网盘如果不开会员就会被 ...

  10. 一个吊打百度网盘的开源神器,还是99年妹子开发的

    文末有干货 "Python高校",马上关注 真爱,请置顶或星标 来源丨开源最前线(ID:OpenSourceTop) 猿妹整编 百度网盘大家都很熟悉,大家都知道百度网盘如果不开会员 ...

最新文章

  1. 12-09关于几种排序方式
  2. about diigo
  3. android 页面翻转进场动画_Android实现翻转及延迟动画效果
  4. sigprocmask理解
  5. C中strcpy函数实现
  6. hdu4403暴力搜索
  7. 你是一个有价值的产品经理吗?
  8. Cascade R-CNN的一些记录
  9. 关于 iOS apps 的启动页设计,再唠叨几句
  10. JS加密算法简单分析
  11. python3 random模块操作
  12. Python练习:百分制到五级制的转换
  13. PHP教程 数据库和MySQL,PHP与MYSQL数据库连接教程分享
  14. verilog学习记(开头篇)
  15. 计算机科学与技术学了会秃头吗,计算机科学与技术vs 信息与计算科学,秃头专业了解一下?...
  16. 服务器 --- 开发框架
  17. 凸优化第四章凸优化问题 4.6广义不等式约束
  18. ToStringBuilder.reflectionToString
  19. HTML5 第六章盒子模型课后作业
  20. c语言报告 列主元lu分解,列主元LU分解法 C++程序

热门文章

  1. P4317-花神的数论题【组合数学】
  2. ZOJ1101-赌徒【二分查找】
  3. Mysql调优你不知道这几点,就太可惜了
  4. JavaFX官方教程(五)之在JavaFX中创建表单
  5. 你知道面试官是如何刷人的吗
  6. JVM内存管理------GC算法精解(复制算法与标记/整理算法)
  7. Java中的ThreadPoolExecutor类
  8. Java NIO总结
  9. 算法六之直接插入排序
  10. 最好的方法,是为你们量身定制!