高并发系统下, 有三把利器 缓存 降级 限流.

  • 缓存: 将常用数据缓存起来, 减少数据库或者磁盘IO
  • 降级: 保护核心系统, 降低非核心业务请求响应
  • 限流: 在某一个时间窗口内对请求进行限速, 保护系统

本文主要介绍限流, 常见限流算法中又分为计数器算法, 漏桶算法, 令牌桶算法.

计数器算法

比较简单, 直接用一个map + counter即可实现. 请求来了, 以IP为key,

查询下之前响应次数, 如果调用次数超出MAX_COUT, 返回失败, 属于简单粗暴型选手.

漏桶算法

请求全部进入漏桶, 漏桶恒定速率输出反馈. 这样可以保证数据传输平滑,

但是无法预防突发大量请求, 一秒来了100个请求, 都要阻塞排队, 从小水管输出数据.

令牌桶算法

令牌桶是以固定速度往桶里存令牌, 例如一秒存1000个令牌, 业务请求来了, 直接从桶里获取令牌响应输出.

跟漏桶的差异在于, 他可以预存令牌, 如果一秒钟来了100个请求, 桶里有100个令牌,

那么可以立刻响应给客户端, 而不是排队输出.

令牌桶的实现

guava中提供了令牌桶的一个封装实现RateLimiter, 可以直接调用, 省的我们自己包装ConcurrentHashMap + Timer.

我们预设的场景是服务器端提供一个API供不同客户端查询, 要限流每个IP每秒只能调用两次该API.

首先要定义一个服务器端的缓存, 定期清理即可, 缓存 IP : 令牌桶

  1. // 根据IP分不同的令牌桶, 每天自动清理缓存
  2. private static LoadingCache<String, RateLimiter> caches = CacheBuilder.newBuilder()
  3. .maximumSize(1000)
  4. .expireAfterWrite(1, TimeUnit.DAYS)
  5. .build(new CacheLoader<String, RateLimiter>() {
  6. @Override
  7. public RateLimiter load(String key) throws Exception {
  8. // 新的IP初始化 (限流每秒两个令牌响应)
  9. return RateLimiter.create(2);
  10. }
  11. });

然后在业务代码中进行限流调用

  1. private static void login(int i) throws ExecutionException {
  2. // 模拟IP的key
  3. String ip = String.valueOf(i).charAt(0) + "";
  4. RateLimiter limiter = caches.get(ip);
  5.  
  6. if (limiter.tryAcquire()) {
  7. System.out.println(i + " success " + new SimpleDateFormat("HH:mm:ss.sss").format(new Date()));
  8. } else {
  9. System.out.println(i + " failed " + new SimpleDateFormat("HH:mm:ss.sss").format(new Date()));
  10. }
  11. }

模拟客户端调用

  1. for (int i = 0; i < 1000; i++) {
  2. // 模拟实际业务请求
  3. Thread.sleep(100);
  4. login(i);
  5. }

完整代码

  1. public class doLimit {
  2.  
  3. // 根据IP分不同的令牌桶, 每天自动清理缓存
  4. private static LoadingCache<String, RateLimiter> caches = CacheBuilder.newBuilder()
  5. .maximumSize(1000)
  6. .expireAfterWrite(1, TimeUnit.DAYS)
  7. .build(new CacheLoader<String, RateLimiter>() {
  8. @Override
  9. public RateLimiter load(String key) throws Exception {
  10. // 新的IP初始化 (限流每秒两个令牌响应)
  11. return RateLimiter.create(2);
  12. }
  13. });
  14.  
  15. public static void main(String[] args) throws InterruptedException, ExecutionException {
  16. for (int i = 0; i < 1000; i++) {
  17. // 模拟实际业务请求
  18. Thread.sleep(100);
  19. login(i);
  20. }
  21. }
  22.  
  23. private static void login(int i) throws ExecutionException {
  24. // 模拟IP的key
  25. String ip = String.valueOf(i).charAt(0) + "";
  26. RateLimiter limiter = caches.get(ip);
  27.  
  28. if (limiter.tryAcquire()) {
  29. System.out.println(i + " success " + new SimpleDateFormat("HH:mm:ss.sss").format(new Date()));
  30. } else {
  31. System.out.println(i + " failed " + new SimpleDateFormat("HH:mm:ss.sss").format(new Date()));
  32. }
  33. }
  34. }

Java 对IP请求进行限流.的更多相关文章

  1. java 模拟http请求,通过流(stream)的方式,发送json数据和文件

    发送端: /** * 以流的方式 * 发送文件和json对象 * * @return */ public static String doPostFileStreamAndJsonObj(String ...

  2. 高可用服务设计之二:Rate limiting 限流与降级

    <高可用服务设计之二:Rate limiting 限流与降级> <nginx限制请求之一:(ngx_http_limit_conn_module)模块> <nginx限制 ...

  3. 限流降级神器,带你解读阿里巴巴开源 Sentinel 实现原理

    Sentinel 是阿里中间件团队开源的,面向分布式服务架构的轻量级高可用流量控制组件,主要以流量为切入点,从流量控制.熔断降级.系统负载保护等多个维度来帮助用户保护服务的稳定性. 大家可能会问:Se ...

  4. 【Distributed】限流技巧

    一.概述 1.1 高并发服务限流特技 1.2 为什么要互联网项目要限流 1.3 高并发限流解决方案 二.限流算法 2.1 计数器 2.2 滑动窗口计数 2.3 令牌桶算法 使用RateLimiter实 ...

  5. 限流10万QPS、跨域、过滤器、令牌桶算法-网关Gateway内容都在这儿

    一.微服务网关Spring Cloud Gateway 1.1 导引 文中内容包含:微服务网关限流10万QPS.跨域.过滤器.令牌桶算法. 在构建微服务系统中,必不可少的技术就是网关了,从早期的Zuu ...

  6. 微服务架构 | 5.2 基于 Sentinel 的服务限流及熔断

    目录 前言 1. Sentinel 基础知识 1.1 Sentinel 的特性 1.2 Sentinel 的组成 1.3 Sentinel 控制台上的 9 个功能 1.4 Sentinel 工作原理 ...

  7. 高并发之 API 接口,分布式,防刷限流,如何做?

    在开发分布式高并发系统时有三把利器用来保护系统:缓存.降级.限流 缓存 缓存的目的是提升系统访问速度和增大系统处理容量 降级 降级是当服务出现问题或者影响到核心流程时,需要暂时屏蔽掉,待高峰或者问题解 ...

  8. WebApiThrottle限流框架使用手册

    阅读目录: 介绍 基于IP全局限流 基于IP的端点限流 基于IP和客户端key的端点限流 IP和客户端key的白名单 IP和客户端key自定义限制频率 端点自定义限制频率 关于被拒请求的计数器 在we ...

  9. 前后端分离djangorestframework——限流频率组件

    频率限制 什么是频率限制 目前我们开发的都是API接口,且是开房的API接口.传给前端来处理的,也就是说,只要有人拿到这个接口,任何人都可以通过这个API接口获取数据,那么像网络爬虫的,请求速度又快, ...

随机推荐

  1. Mongodb基础与入门

    一:基本了解                1. 特点                        基于分布式文件存储的NoSql数据库.能为WEB应用提供可扩展的高性能数据存储解决方案.      ...

  2. mac中配置jdk环境

  3. Python数据结构之四——set(集合)

    Python版本:3.6.2  操作系统:Windows  作者:SmallWZQ 经过几天的回顾和学习,我终于把Python 3.x中的基础知识介绍好啦.下面将要继续什么呢?让我想想先~~~嗯,还是 ...

  4. 《精通android网络开发》--使用Socket实现数据通信

    No1: 网络传输应用通常使用TCP.IP或UDP这三种协议实现数据传输.在传输数据的过程中,需要通过一个双向的通信连接实现数据的交互.在这个传输过程中,通常将这个双向链路的一端称为Socket,一个 ...

  5. AndroidDevTools

    收集整理Android开发所需的Android SDK.开发中用到的工具.Android开发教程.Android设计规范,免费的设计素材等. 欢迎大家推荐自己在Android开发过程中用的好用的工具. ...

  6. 导出CSV格式文件,用Excel打开乱码的解决办法

    导出CSV格式文件,用Excel打开乱码的解决办法 1.治标不治本的办法 将导出CSV数据文件用记事本打开,然后另存为"ANSI"编码格式,再用Excel打开,乱码解决. 但是,这 ...

  7. VTK显示mhd,mha格式文件

    下一篇文章将详细介绍mhd,mha文件的构成以及如何制作void renderMhd () { // read input image vtkSmartPointer<vtkMetaImageR ...

  8. The currently displayed page contains invalid values.

    1 错误描述 2 错误原因 3 解决办法

  9. Linux显示历史记录

    Linux显示历史记录 youhaidong@youhaidong-ThinkPad-Edge-E545:~$ history 1 uname -a 2 lsusb 3 df -h 4 ps -A 5 ...

  10. JSP自定义标签配置

    JSP自定义标签配置 JSP自定义标签 <taglib>         <taglib-uri>/WEB-INF/you.tld</taglib-uri>     ...