SpringBooot如何进行限流

目录

1.什么是限流?

2.为什么要进行限流?

3.SpringBoot限流实例


1.什么是限流?

        限流可以认为服务降级的一种,限流就是限制系统的输入和输出流量已达到保护系统的目的。一般来说系统的吞吐量是可以被测算的,为了保证系统的稳定运行,一旦达到的需要限制的阈值,就需要限制流量并采取一些措施以完成限制流量的目的。

2.为什么要进行限流?

        互联网系统通常都要面对大并发大流量的请求,在突发情况下(最常见的场景就是秒杀、抢购),瞬时大流量会直接将系统打垮,无法对外提供服务。那为了防止出现这种情况最常见的解决方案之一就是限流,当请求达到一定的并发数或速率,就进行等待、排队、降级、拒绝服务等。

        例如:12306购票系统,在面对高并发的情况下,就是采用了限流。在流量高峰期间经常会出现提示语;"当前排队人数较多,请稍后再试!"

3.SpringBoot限流实例

        3.1 加上依赖

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-aop</artifactId>
        </dependency>
        <dependency>
            <groupId>com.google.guava</groupId>
            <artifactId>guava</artifactId>
            <version>30.1-jre</version>
        </dependency>

        3.2 自定义限流注解

@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD})
@Documented
public @interface Limit {
    /**
     * 资源的key,唯一
     * 作用:不同的接口,不同的流量控制
     */
    String key() default "";

    /**
     * 最多的访问限制次数
     */
    double permitsPerSecond () ;

    /**
     * 获取令牌最大等待时间
     */
    long timeout();

    /**
     * 获取令牌最大等待时间,单位(例:分钟/秒/毫秒) 默认:毫秒
     */
    TimeUnit timeunit() default TimeUnit.MILLISECONDS;

    /**
     * 得不到令牌的提示语
     */
    String msg() default "系统繁忙,请稍后再试.";
}

        3.3 使用AOP切面拦截限流注解

@Slf4j
@Aspect
@Component
public class LimitAop {
    /**
     * 不同的接口,不同的流量控制
     * map的key为 Limiter.key
     */
    private final Map<String, RateLimiter> limitMap = Maps.newConcurrentMap();

    @Around("@annotation(cn.mpy.aopguava.common.Limit)")
    public Object around(ProceedingJoinPoint joinPoint) throws Throwable{
        MethodSignature signature = (MethodSignature) joinPoint.getSignature();
        Method method = signature.getMethod();
        //拿limit的注解
        Limit limit = method.getAnnotation(Limit.class);
        if (limit != null) {
            //key作用:不同的接口,不同的流量控制
            String key=limit.key();
            RateLimiter rateLimiter = null;
            //验证缓存是否有命中key
            if (!limitMap.containsKey(key)) {
                // 创建令牌桶
                rateLimiter = RateLimiter.create(limit.permitsPerSecond());
                limitMap.put(key, rateLimiter);
                log.info("新建了令牌桶={},容量={}",key,limit.permitsPerSecond());
            }
            rateLimiter = limitMap.get(key);
            // 拿令牌
            boolean acquire = rateLimiter.tryAcquire(limit.timeout(), limit.timeunit());
            // 拿不到命令,直接返回异常提示
            if (!acquire) {
                log.debug("令牌桶={},获取令牌失败",key);
                return limit.msg();
            }
        }
        return joinPoint.proceed();
    }

}

         3.4 给需要限流的接口加上注解

@Slf4j
@RestController
@RequestMapping("/limit")
public class LimitController {
    @GetMapping("/test2")
    @Limit(key = "limit2", permitsPerSecond = 1, timeout = 500, timeunit = TimeUnit.MILLISECONDS,msg = "当前排队人数较多,请稍后再试!")
    public String limit2() {
        log.info("令牌桶limit2获取令牌成功");
        return "ok";
    }


    @GetMapping("/test3")
    @Limit(key = "limit3", permitsPerSecond = 2, timeout = 500, timeunit = TimeUnit.MILLISECONDS,msg = "系统繁忙,请稍后再试!")
    public String limit3() {
        log.info("令牌桶limit3获取令牌成功");
        return "ok";
    }
}

         3.5 测试

        用浏览器访问http://127.0.0.1:8080/limit/test2或者http://127.0.0.1:8080/limit/test3快速刷新

            正常,没有被限流:

                      ​​​​​​

           接口被限流:

        ​​​​​​​               

  • 2
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区管理系统小区
Spring Boot是一个开源的Java框架,可以简化基于Java的应用程序的开发和部署。它提供了一种快速开发、易于扩展和部署的方式,使得开发者可以更加专注于业务逻辑的实现。 而MyBatis Plus是一个基于MyBatis的增强工具,它提供了许多强大的功能和特性,能够更加方便地与数据库进行交互。相对于原生的MyBatis,MyBatis Plus具有更加简洁和易用的API,使得开发者可以更加高效地编写SQL语句并进行数据库操作。 Spring Boot与MyBatis Plus的结合可以让Java开发者更加轻松地构建敏捷的web应用。Spring Boot提供了自动化配置和集成多个常用的开发工具和框架,使得我们可以快速搭建一个可运行的应用。而MyBatis Plus则提供了更好的ORM(Object Relational Mapping)支持,可以简化SQL的编写和数据操作。 使用Spring Boot和MyBatis Plus的好处是多方面的。首先,可以极大地提高开发效率,因为Spring Boot提供了自动化配置和集成,而MyBatis Plus则提供了更加便捷的数据库操作方法。其次,两者都遵循同一种理念,即约定优于配置,使得开发者可以轻松编写出简洁而高效的代码。此外,Spring Boot和MyBatis Plus都具有很好的社区支持和文档资料,可以方便地找到解决问题的方法和参考。 综上所述,Spring Boot与MyBatis Plus的结合可以大大提高Java开发的效率和便捷性,使得我们可以更加专注于业务逻辑的实现,从而快速构建出高质量的web应用。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

@insist123

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值