如何通过限流算法防止系统过载

news2025/1/12 12:00:46

限流算法,顾名思义,就是指对流量进行控制的算法,因此也常被称为流控算法。

我们在日常生活中,就有很多限流的例子,比如地铁站在早高峰的时候,会利用围栏让乘客们有序排队,限制队伍行进的速度,避免大家一拥而上;再比如在疫情期间,很多景点会按时段限制售卖的门票数量,避免同一时间在景区的游客太多等等。

对于 Server 服务而言,单位时间内能承载的请求也是存在容量上限的,我们也需要通过一些策略,控制请求数量多少,实现对流量的控制,虽然,限流为了保证一部分的请求流量可以得到正常的响应,一定会导致部分请求响应速度下降或者直接被拒绝,但是相比于全部的请求都不能得到响应,系统直接崩溃的情况,限流还是要好得多。

本篇内容主要介绍:业务中的限流场景、限流算法、限流值的确认、不仅仅限流


文章目录

    • 一、业务中的限流场景
        • 1、限流算法介绍
        • 2、突发流量
        • 3、恶意流量
        • 4、业务本身需要
    • 二、限流算法
        • 1、固定窗口计数器
        • 2、滑动窗口计数器
        • 3、Leaky Bucket 漏桶 - As a Meter Version
        • 4、Leaky Bucket 漏桶 - As a Queue Version
        • 5、Token Bucket 令牌桶
    • 三、限流相关问题
        • 1、限流值的确认
        • 2、不仅仅限流


一、业务中的限流场景

1、限流算法介绍

限流算法,顾名思义,就是指对流量进行控制的算法,因此也常被称为流控算法。

我们在日常生活中,就有很多限流的例子,比如地铁站在早高峰的时候,会利用围栏让乘客们有序排队,限制队伍行进的速度,避免大家一拥而上;再比如在疫情期间,很多景点会按时段限制售卖的门票数量,避免同一时间在景区的游客太多等等。

对于 Server 服务而言,单位时间内能承载的请求也是存在容量上限的,我们也需要通过一些策略,控制请求数量多少,实现对流量的控制,虽然,限流为了保证一部分的请求流量可以得到正常的响应,一定会导致部分请求响应速度下降或者直接被拒绝,但是相比于全部的请求都不能得到响应,系统直接崩溃的情况,限流还是要好得多。

限流与熔断经常被人弄混,博主认为它们最大的区别在于限流主要在 Server 实现,而熔断主要在 Client 实现,当然了,一个服务既可以充当 Server 也可以充当 Client,这也是让限流与熔断同时存在一个服务中,这两个概念才容易被混淆。

业务中的典型的限流场景主要分为三种:

  • 突发流量
  • 恶意流量
  • 业务本身需要

2、突发流量

突发流量是我们需要限流的主要场景之一。当我们后端服务处理能力有限,面对业务流量突然激增,即突发流量时,很容易出现服务器被打垮的情况。

如我们常见的双十一,京东 618 这些整点秒杀的业务,12306 这些都会出现某段时间面临着大量的流量流入的情况。

在这些情况下,除了提供更好的弹性伸缩的能力,以及在已经能预测的前提下提前准备更多的资源,我们还能做的一件事就是利用限流来保护服务,即使拒绝了一部分请求,至少也让剩下的请求可以正常被响应。

3、恶意流量

除了突发流量,限流有的时候也是出于安全性的考虑。网络世界有其凶险的地方,所有暴露出去的API都有可能面对非正常业务的请求。

比如我们常见的各种各样的网络爬虫,或者恶意的流量攻击网站等等,都会产生大量的恶意流量。面对我们服务对外暴露接口的大规模疯狂调用,很有可能也会可能导致服务崩溃,在很多时候也会导致我们需要的计算成本飙升,比如云计算的场景下。

4、业务本身需要

还有一种业务本身需要的场景,这种场景也十分常见,比如云服务平台根据不同套餐等级,需要对不同服务流量限制时,也是需要采取限流算法的。


二、限流算法

终于到了正题,我们这里将介绍 4 种限流算法:分别是 固定窗口计数器、滑动窗口计数器、Leaky Bucket 漏桶、Token Bucket令牌桶

1、固定窗口计数器

规定我们单位时间处理的请求数量。比如我们规定我们的一个接口一分钟只能访问10次的话。使用固定窗口计数器算法的话可以这样实现:给定一个变量counter来记录处理的请求数量,当1分钟之内处理一个请求之后counter+1,1分钟之内的如果counter=100的话,后续的请求就会被全部拒绝。等到 1分钟结束后,将counter回归成0,重新开始计数(ps:只要过了一个周期就讲counter回归成0)。

image-20230117115505429

这种限流算法无法保证限流速率,因而无法保证突然激增的流量。比如我们限制一个接口一分钟只能访问10次的话,前半分钟一个请求没有接收,后半分钟接收了10个请求。

# 具体实现

package com.lizhengi.limiter;

import java.util.concurrent.*;
import java.util.concurrent.atomic.AtomicInteger;

/**
 * @author liziheng
 * @version 1.0.0
 * @description 固定窗口计数器
 * @date 2023-01-17 2:20 下午
 **/
public class FixedWindowCounterLimiter {

    /**
     * 限流阈值
     */
    private final int limit;

    /**
     * 计数器
     */
    private final AtomicInteger count;

    /**
     * 固定窗口计数器
     *
     * @param windowSize 时间窗口大小, Unit: s
     * @param limit      限流阈值
     */
    public FixedWindowCounterLimiter(int windowSize, int limit) {
        this.limit = limit;
        count = new AtomicInteger(0);

        // 通过线程池启动一个线程, 定时清除计数器值
        ExecutorService threadPool = new ThreadPoolExecutor(1, 1, 1L, TimeUnit.SECONDS,
                new LinkedBlockingQueue<>(3), Executors.defaultThreadFactory(), new ThreadPoolExecutor.AbortPolicy());
        threadPool.execute(() -> {
            // noinspection InfiniteLoopStatement
            while (true) {
                try {
                    Thread.sleep(windowSize * 1000L);
                } catch (InterruptedException e) {
                    System.out.println("Happen Exception: " + e.getMessage());
                }
                count.set(0);
            }
        });

    }

    public boolean tryAcquire() {
        int num = count.incrementAndGet();
        // 以达到当前窗口的请求阈值
        return num <= limit;
    }
}

# 测试代码

package com.lizhengi.limiter;

import org.junit.Test;

/**
 * @author liziheng
 * @version 1.0.0
 * @description 固定窗口计数器测试
 * @date 2023-01-17 2:22 下午
 **/
public class FixedWindowCounterLimiterTest {

    @Test
    public void test() throws InterruptedException {
        // 请求总数 通过数 被限流数
        int allNum, passNum = 0, blockNum = 0;
        // 限流配置 : 2s 内只允许通过 5个 !
        FixedWindowCounterLimiter rateLimiter = new FixedWindowCounterLimiter(2, 5);

        // 限流测试 1 - 请求总数 设置 3 次
        allNum = 3;
        //模拟连续请求
        for (int i = 0; i < allNum; i++) {
            if (rateLimiter.tryAcquire()) {
                passNum++;
            } else {
                blockNum++;
            }
        }
        System.out.println("请求总数: " + allNum + ", 通过数: " + passNum + ", 被限流数: " + blockNum);

        // 延时以准备下一次测试
        Thread.sleep(5000);
        // 限流测试 2 - 请求总数 设置 14 次
        allNum = 14;
        passNum = blockNum = 0;
        //模拟连续请求
        for (int i = 0; i < allNum; i++) {
            if (rateLimiter.tryAcquire()) {
                passNum++;
            } else {
                blockNum++;
            }
        }
        System.out.println("请求总数: " + allNum + ", 通过数: " + passNum + ", 被限流数: " + blockNum);
    }
}

# 测试结果

请求总数: 3, 通过数: 3, 被限流数: 0
请求总数: 14, 通过数: 5, 被限流数: 9

2、滑动窗口计数器

算的上是固定窗口计数器算法的升级版。滑动窗口计数器算法相比于固定窗口计数器算法的优化在于:它把时间以一定比例分片。例如我们的借口限流每分钟处理60个请求,我们可以把 1 分钟分为60个窗口。每隔1秒移动一次,每个窗口一秒只能处理 不大于 60(请求数)/60(窗口数) 的请求, 如果当前窗口的请求计数总和超过了限制的数量的话就不再处理其他请求。

image-20230117115613214

很显然:当滑动窗口的格子划分的越多,滑动窗口的滚动就越平滑,限流的统计就会越精确。

# 具体实现

package com.lizhengi.limiter;

import java.util.Arrays;

/**
 * @author liziheng
 * @version 1.0.0
 * @description 滑动时间窗口计数器
 * @date 2023-01-17 3:02 下午
 **/
public class SlidingWindowCounterLimiter {

    /**
     * 用于统计的子窗口数量,默认为10
     */
    private final int slotNum;

    /**
     * 子窗口的时间长度, Unit: ms
     */
    private final int slotTime;

    /**
     * 限流阈值
     */
    private final int limit;

    /**
     * 存放子窗口统计结果的数组
     * note: counters[0]记为数组左边, counters[size-1]记为数组右边
     */
    private final int[] counters;

    private long lastTime;

    public SlidingWindowCounterLimiter(int windowSize, int limit) {
        this(windowSize, limit, 10);
    }

    public SlidingWindowCounterLimiter(int windowSize, int limit, int slotNum) {
        this.limit = limit;
        this.slotNum = slotNum;

        this.counters = new int[slotNum];
        // 计算子窗口的时间长度: 时间窗口 / 子窗口数量
        this.slotTime = windowSize * 1000 / slotNum;
        this.lastTime = System.currentTimeMillis();
    }

    public synchronized boolean tryAcquire() {
        long currentTime = System.currentTimeMillis();
        // 计算滑动数, 子窗口统计时所对应的时间范围为左闭右开区间, 即[a,b)
        int slideNum = (int) Math.floor((currentTime - lastTime) * 1.0 / slotTime);
        // 滑动窗口
        slideWindow(slideNum);
        // 统计滑动后的数组之和
        int sum = Arrays.stream(counters).sum();

        // 以达到当前时间窗口的请求阈值, 故被限流直接返回false
        if (sum > limit) {
            return false;
        } else {    // 未达到限流, 故返回true
            counters[slotNum - 1]++;
            return true;
        }
    }

    /**
     * 将数组元素全部向左移动num个位置
     *
     * @param num 移动位置数目
     */
    private void slideWindow(int num) {
        if (num == 0) {
            return;
        }

        // 数组中所有元素都会被移出, 故直接全部清零
        if (num >= slotNum) {
            Arrays.fill(counters, 0);
        } else {
            // 对于a[0]~a[num-1]而言, 向左移动num个位置后, 则直接被移出了
            // 故从a[num]开始移动即可
            for (int index = num; index < slotNum; index++) {
                // 计算a[index]元素向左移动num个位置后的新位置索引
                int newIndex = index - num;
                counters[newIndex] = counters[index];
                counters[index] = 0;
            }
        }
        // 更新时间
        lastTime = lastTime + (long) num * slotTime;
    }

}

# 测试代码

package com.lizhengi.limiter;

import org.junit.Test;

/**
 * @author liziheng
 * @version 1.0.0
 * @description 滑动时间窗口计数器测试
 * @date 2023-01-17 3:04 下午
 **/
public class SlidingWindowCounterLimiterTest {

    @Test
    public void test() throws InterruptedException {
        // 请求总数 通过数 被限流数
        int allNum, passNum = 0, blockNum = 0;
        // 限流配置 : 2s 内只允许通过 5个 !
        SlidingWindowCounterLimiter rateLimiter = new SlidingWindowCounterLimiter(2, 5);

        // 限流测试 1 - 请求总数 设置 3 次
        allNum = 3;
        //模拟连续请求
        for (int i = 0; i < allNum; i++) {
            if (rateLimiter.tryAcquire()) {
                passNum++;
            } else {
                blockNum++;
            }
        }
        System.out.println("请求总数: " + allNum + ", 通过数: " + passNum + ", 被限流数: " + blockNum);

        // 延时以准备下一次测试
        Thread.sleep(5000);
        // 限流测试 2 - 请求总数 设置 14 次
        allNum = 14;
        passNum = blockNum = 0;
        //模拟连续请求
        for (int i = 0; i < allNum; i++) {
            if (rateLimiter.tryAcquire()) {
                passNum++;
            } else {
                blockNum++;
            }
        }
        System.out.println("请求总数: " + allNum + ", 通过数: " + passNum + ", 被限流数: " + blockNum);
    }
}

# 测试结果

请求总数: 3, 通过数: 3, 被限流数: 0
请求总数: 14, 通过数: 6, 被限流数: 8

3、Leaky Bucket 漏桶 - As a Meter Version

我们可以把发请求的动作比作成注水到桶中,我们处理请求的过程可以比喻为漏桶漏水。我们往桶中以任意速率流入水,以一定速率流出水。当水超过桶流量则丢弃,因为桶容量是不变的,保证了整体的速率。如果想要实现这个算法的话也很简单,准备一个队列用来保存请求,然后我们定期从队列中拿请求来执行就好了。

image-20230117115939370

# 具体实现

package com.lizhengi.limiter;

/**
 * @author liziheng
 * @version 1.0.0
 * @description
 * @date 2023-01-17 3:33 下午
 **/
public class LeakyBucketLimiter1 {

    /**
     * 桶容量, Unit: 个
     */
    private final long capacity;

    /**
     * 出水速率, Unit: 个/秒
     */
    private final long rate;

    /**
     * 桶的当前水量
     */
    private long water;

    /**
     * 上次时间
     */
    private long lastTime;

    public LeakyBucketLimiter1(long capacity, long rate) {
        this.capacity = capacity;
        this.rate = rate;
        this.water = 0;
        this.lastTime = System.currentTimeMillis();
    }

    public synchronized boolean tryAcquire() {
        // 获取当前时间
        long currentTime = System.currentTimeMillis();
        // 计算流出的水量: (当前时间-上次时间) * 出水速率
        long outWater = (currentTime - lastTime) / 1000 * rate;
        // 计算水量: 桶的当前水量 - 流出的水量
        water = Math.max(0, water - outWater);
        // 更新时间
        lastTime = currentTime;

        // 当前水量 小于 桶容量, 则请求放行, 返回true
        if (water < capacity) {
            water++;
            return true;
        } else {
            // 当前水量 不小于 桶容量, 则进行限流, 返回false
            return false;
        }
    }

}

# 测试代码

package com.lizhengi.limiter;

import org.junit.Test;

/**
 * @author liziheng
 * @version 1.0.0
 * @description
 * @date 2023-01-17 3:34 下午
 **/
public class LeakyBucketLimiter1Test {

    @Test
    public void test() throws InterruptedException {
        // 请求总数 通过数 被限流数
        int allNum, passNum = 0, blockNum = 0;
        // 漏桶配置, 桶容量:5个, 出水率: 1个/秒
        LeakyBucketLimiter1 rateLimiter = new LeakyBucketLimiter1(5, 1);

        // 限流测试 1 - 请求总数 设置 3 次
        allNum = 3;
        //模拟连续请求
        for (int i = 0; i < allNum; i++) {
            if (rateLimiter.tryAcquire()) {
                passNum++;
            } else {
                blockNum++;
            }
        }
        System.out.println("请求总数: " + allNum + ", 通过数: " + passNum + ", 被限流数: " + blockNum);

        // 延时以准备下一次测试
        Thread.sleep(5000);
        // 限流测试 2 - 请求总数 设置 14 次
        allNum = 14;
        passNum = blockNum = 0;
        //模拟连续请求
        for (int i = 0; i < allNum; i++) {
            if (rateLimiter.tryAcquire()) {
                passNum++;
            } else {
                blockNum++;
            }
        }
        System.out.println("请求总数: " + allNum + ", 通过数: " + passNum + ", 被限流数: " + blockNum);
    }
}

# 测试结果

请求总数: 3, 通过数: 3, 被限流数: 0
请求总数: 14, 通过数: 5, 被限流数: 9

4、Leaky Bucket 漏桶 - As a Queue Version

在 As a Meter Version 版本的漏桶中,当桶中水未满,请求即会直接被放行。而在漏桶的另外一个版本 As a Queue Version 中,如果桶中水未满,则该请求将会被暂时存储在桶中。然后以漏桶固定的出水速率对桶中存储的请求依次放行。对比两个版本的漏桶算法不难看出,As a Meter Version 版本的漏桶算法可以应对、处理突发流量,只要桶中尚有足够空余即可立即放行请求;而对于 As a Queue Version 版本的漏桶,其只会以固定速率放行请求,无法充分利用后续系统的处理能力。

# 具体实现

package com.lizhengi.limiter;

import lombok.AllArgsConstructor;
import org.apache.commons.lang3.concurrent.BasicThreadFactory;

import java.time.LocalTime;
import java.time.format.DateTimeFormatter;
import java.util.concurrent.*;

/**
 * @author liziheng
 * @version 1.0.0
 * @description
 * @date 2023-01-17 3:38 下午
 **/
public class LeakyBucketLimiter2 {

    /**
     * 阻塞队列, 用于存储用户请求
     */
    private final ArrayBlockingQueue<UserRequest> queue;

    /**
     * @param capacity 桶容量, Unit: 个
     * @param rate     出水速率, Unit: 个/秒
     */
    public LeakyBucketLimiter2(int capacity, long rate) {
        // 根据桶容量构建有界队列
        queue = new ArrayBlockingQueue<>(capacity);

        // 定时任务线程池, 用于以指定速率rate从阻塞队列中获取用户请求进行放行、处理
        ScheduledExecutorService threadPool = new ScheduledThreadPoolExecutor(1, new BasicThreadFactory.
                Builder().namingPattern("example-schedule-pool-%d").daemon(true).build());
        // 根据出水速率rate计算从阻塞队列获取用户请求的周期, Unit: ms
        long period = 1000 / rate;
        threadPool.scheduleAtFixedRate(getTask(), 0, period, TimeUnit.MILLISECONDS);
    }

    public boolean tryAcquire(UserRequest userRequest) {
        // 添加失败表示用户请求被限流, 则返回false
        return queue.offer(userRequest);
    }

    private Runnable getTask() {
        return () -> {
            // 从阻塞队列获取用户请求
            UserRequest userRequest = queue.poll();
            if (userRequest != null) {
                userRequest.handle();
            }
        };
    }

    /**
     * 用户请求
     */
    @AllArgsConstructor
    public static class UserRequest {

        private static final DateTimeFormatter FORMATTER = DateTimeFormatter.ofPattern("HH:mm:ss.SSS");

        private String name;

        public void handle() {
            String timeStr = FORMATTER.format(LocalTime.now());
            String msg = "<" + timeStr + "> " + name + " 开始处理";
            System.out.println(msg);
        }
    }

}

# 代码测试

package com.lizhengi.limiter;

import org.junit.Test;

/**
 * @author liziheng
 * @version 1.0.0
 * @description
 * @date 2023-01-17 3:46 下午
 **/
public class LeakyBucketLimiter2Test {

    @Test
    public void test() throws InterruptedException {
        // 请求总数 通过数 被限流数
        int allNum, passNum = 0, blockNum = 0;
        // 漏桶配置, 桶容量:5个, 出水率: 2个/秒
        LeakyBucketLimiter2 rateLimiter = new LeakyBucketLimiter2(5, 2);

        // 限流测试 1 - 请求总数 设置 7 次
        allNum = 7;
        // 模拟连续请求
        for(int i=1; i<=allNum; i++) {
            // 构建用户请求
            String name = "用户请求:" + i;
            LeakyBucketLimiter2.UserRequest userRequest = new LeakyBucketLimiter2.UserRequest(name);

            if( rateLimiter.tryAcquire( userRequest ) ) {
                passNum++;
            }else{
                blockNum++;
            }
        }
        System.out.println("请求总数: "+allNum+", 通过数: "+passNum+", 被限流数: "+blockNum);

        // 延时等待
        Thread.sleep(120*1000);
    }
}

# 测试结果

请求总数: 7, 通过数: 5, 被限流数: 2
<15:48:21.542> 用户请求:1 开始处理
<15:48:22.032> 用户请求:2 开始处理
<15:48:22.532> 用户请求:3 开始处理
<15:48:23.032> 用户请求:4 开始处理
<15:48:23.533> 用户请求:5 开始处理

5、Token Bucket 令牌桶

令牌桶算法也比较简单。和漏桶算法算法一样,我们的主角还是桶(这限流算法和桶过不去啊)。不过现在桶里装的是令牌了,请求在被处理之前需要拿到一个令牌,请求处理完毕之后将这个令牌丢弃(删除)。我们根据限流大小,按照一定的速率往桶里添加令牌。

image-20230117120058362# 具体实现

package com.lizhengi.limiter;

/**
 * @author liziheng
 * @version 1.0.0
 * @description
 * @date 2023-01-17 3:52 下午
 **/
public class TokenBucketLimiter {

    /**
     * 桶容量, Unit: 个
     */
    private final long capacity;

    /**
     * 令牌生成速率, Unit: 个/秒
     */
    private final long rate;

    /**
     * 桶当前的令牌数量
     */
    private long tokens;

    /**
     * 上次时间
     */
    private long lastTime;

    public TokenBucketLimiter(long capacity, long rate) {
        this.capacity = capacity;
        this.rate = rate;
        this.tokens = capacity;
        this.lastTime = System.currentTimeMillis();
    }

    public synchronized boolean tryAcquire() {
        // 获取当前时间
        long currentTime = System.currentTimeMillis();
        // 计算生成的令牌数量: (当前时间-上次时间) * 令牌生成速率
        long newTokenNum = (currentTime - lastTime) / 1000 * rate;
        // 计算令牌数量: 桶当前的令牌数量 + 生成的令牌数量
        tokens = Math.min(capacity, tokens + newTokenNum);
        // 更新时间
        lastTime = currentTime;

        // 桶中仍有令牌, 则请求放行, 返回true
        if (tokens > 0) {
            tokens--;
            return true;
        } else {
            // 桶中没有令牌, 则进行限流, 返回false
            return false;
        }
    }
}

# 测试代码

package com.lizhengi.limiter;

import org.junit.Test;

/**
 * @author liziheng
 * @version 1.0.0
 * @description
 * @date 2023-01-17 4:01 下午
 **/
public class TokenBucketLimiterTest {
    @Test
    public void test() throws InterruptedException {
        // 请求总数 通过数 被限流数
        int allNum, passNum = 0, blockNum = 0;
        // 令牌桶配置, 桶容量:5个, 令牌生成速率: 1个/秒
        TokenBucketLimiter rateLimiter = new TokenBucketLimiter(5, 1);

        // 限流测试 1 - 请求总数 设置 3 次
        allNum = 3;
        //模拟连续请求
        for (int i = 0; i < allNum; i++) {
            if (rateLimiter.tryAcquire()) {
                passNum++;
            } else {
                blockNum++;
            }
        }
        System.out.println("请求总数: " + allNum + ", 通过数: " + passNum + ", 被限流数: " + blockNum);

        // 延时以准备下一次测试
        Thread.sleep(5000);
        // 限流测试 2 - 请求总数 设置 14 次
        allNum = 14;
        passNum = blockNum = 0;
        //模拟连续请求
        for (int i = 0; i < allNum; i++) {
            if (rateLimiter.tryAcquire()) {
                passNum++;
            } else {
                blockNum++;
            }
        }
        System.out.println("请求总数: " + allNum + ", 通过数: " + passNum + ", 被限流数: " + blockNum);
    }
}

# 测试结果

请求总数: 3, 通过数: 3, 被限流数: 0
请求总数: 14, 通过数: 5, 被限流数: 9


三、限流相关问题

1、限流值的确认

正确的值,才能起到效果;限流多了等于没限,少了则会影响服务利用效率

对于核心服务限流的值可以通过以下方法来设置合理的值

  • 观察评估:通过CAT大盘,可以观察到服务的平时调用量,QPS和各个调用方。
  • 压测摸底:通过quake平台,可以压测核心服务的支持的最大QPS。
  • 场景分析:通过分析各业务调用场景,评估一个合理的值。

2、不仅仅限流

限流作为系统稳定性保障的有效措施之一,常常与重试、降级、熔断等作为组合方法一起使用。

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

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

相关文章

扫雷——“C”

各位uu们我又来啦&#xff0c;今天&#xff0c;小雅兰给大家介绍的又是一个小游戏&#xff0c;就是扫雷这款游戏&#xff0c;这个游戏和我昨天给大家介绍的三子棋游戏有异曲同工之妙&#xff0c;相信很多人都玩过&#xff0c;话不多说&#xff0c;我们进入正题吧. 《扫雷》是一…

【学习笔记】【Pytorch】十七、模型测试套路

【学习笔记】【Pytorch】十七、模型测试套路一、内容概述二、模型测试套路代码实现一、内容概述 利用已经训练好的模型&#xff0c;然后给它提供输入&#xff0c;判断输出是否正确&#xff0c;即模型的应用测试。 在模型测试也会有一些坑&#xff1a; 神经网络的输入一般是4…

【错误记录】Kotlin 代码编译时报错 ( Variable ‘name‘ must be initialized | 初始化块定义在所有属性之后 )

文章目录一、报错信息二、问题分析三、解决方案 ( 初始化块定义在所有属性之后 )一、报错信息 在 Kotlin 中 , init 初始化块 要 定义在所有成员属性之后 ; 如果在 init 初始化块 中 , 使用到了 成员属性 , 有可能出现 编译时报错信息 ; 报错代码示例 : class Hello{init {va…

seata安装及配置

1.下载 下载地址&#xff1a;https://github.com/seata/seata/tags 本文选用seata-1.4.2版 2.解压 tar -zxvf seata-server-1.4.2.tar.gz 3. 初始化数据库 登录mysql&#xff0c;然后创建数据库和数据表&#xff1a; -- -------------------------------- The script used…

【Java】【系列篇】【Spring源码解析】【三】【体系】【Environment体系】

整体结构图 本篇文章仅作简单了解&#xff0c;详细还等到Springboot系列里面详解PropertyResolver 作用 用于针对任何基础源解析属性(Property)的接口 方法解析 // 查看规定指定的key是否有对应的value 对应key的值是null的话也算是不能解析 boolean containsProperty(Stri…

持续丰富营销玩法 东风标致408X引领品牌向上焕新

1月5日&#xff0c;东风标致408X首秀——XSHOW开演&#xff0c;标致全球战略车型408X正式在中国亮相&#xff0c;定位为“新法式无界座驾”&#xff0c;它是东风标致全面向电动化、智能化、网联化的发展的一款汽车&#xff0c;也是引领东风标致向上焕新的一款全新车型。作为东风…

十五天学会Autodesk Inventor,看完这一系列就够了(终章),答疑

众所周知&#xff0c;Autocad是一款用于二维绘图、详细绘制、设计文档和基本三维设计&#xff0c;现已经成为国际上广为流行的绘图工具。Autodesk Inventor软件也是美国AutoDesk公司推出的三维可视化实体模拟软件。因为很多人都熟悉Autocad&#xff0c;所以再学习Inventor&…

python SciPy 优化器

SciPy 优化器SciPy 的 optimize 模块提供了常用的最优化算法函数实现&#xff0c;我们可以直接调用这些函数完成我们的优化问题&#xff0c;比如查找函数的最小值或方程的根等。NumPy 能够找到多项式和线性方程的根&#xff0c;但它无法找到非线性方程的根&#xff0c;如下所示…

工具类库 Hutool介绍与使用(请记得收藏)

工具类库 Hutool介绍 Hutool是一个小而全的Java工具类库&#xff0c;通过静态方法封装&#xff0c;降低相关API的学习成本&#xff0c;提高工作效率&#xff0c;使Java拥有函数式语言般的优雅&#xff0c;让Java语言也可以“甜甜的”。Hutool中的工具方法来自于每个用户的精雕…

VMware vCenter上用OVF模板部署虚拟机

前言 在我们项目操作过程中&#xff0c;使用VMware vCenter系统&#xff0c;经常会出现使用原有部署环境来部署虚拟机&#xff0c;供项目组使用&#xff0c;此时我们克隆虚拟机就要用到OVF模板来克隆操作&#xff0c;这是一个非常实用的功能。 一、打开Vcenter&#xff0c;选定…

Redis学习笔记2_数据结构

Redis数据结构Redis数据结构二、数据结构2.1Redis核心对象2.2底层数据结构2.2.1 SDS-simple dynamic stringsds内存布局sds的操作为什么使用SDS&#xff0c;SDS的优势&#xff1f;2.2.2 listlist内存布局2.2.3 dictdict内存布局2.2.4 zskiplistzskiplist内存布局2.2.5 intsetin…

前端实现登录拼图验证

前言 不知各位朋友现在在 web 端进行登录的时候有没有注意一个变化&#xff0c;以前登录的时候是直接账号密码通过就可以直接登录&#xff0c;再后来图形验证码&#xff0c;数字结果运算验证&#xff0c;到现在的拼图验证。这一系列的转变都是为了防止机器操作&#xff0c;但对…

Python 第六章 函数

6.1函数的定义和调用6.1.1定义函数格式&#xff1a;def 函数名 ([参数列表]):["""文档字符串"""]函数体[return 语句]6.1.2函数调用格式&#xff1a;函数名([参数列表])python中函数可以嵌套定义例如&#xff1a;def add_modify(a,b):resultabpr…

Vue3响应式原理解析

前言 今年上半年开始&#xff0c;自己开始在新项目中使用 Vue3 进行开发&#xff0c;相比较于 Vue2 来说&#xff0c;最大的变化就是 composition Api 代替了之前的 options Api&#xff0c;更像是 React Hooks 函数式组件的编程方式。 Vue3相对于Vue2响应式原理也发生了变化…

vue日期组件el-date-picker中更改默认日期格式并且实时显示的方法

在项目中有一个需求是这样的,要求实时显示他的当前默认时间,并且不能修改 使用了默认:default-value"currentTime"属性之后,新增的时候会报错,前端与后端传递的数据不匹配 因为默认时间被new date() 解析之后返回的数据是默认时间形式的,格式不符 方法如下: 第一步&a…

Elasticsearch入门 - Mac上Elasticsearch和Kibana的安装运行与简单使用

文章目录一&#xff0c;Mac上Elasticsearch和Kibana的安装1.1 环境与下载1.2 安装与运行1.3 问题1.3.1 elasticsearch安装后其他机器不能访问1.3.2 kibana安装后其他机器不能访问二&#xff0c;Elasticsearch在Kibana的常见命令2.1 查看集群的健康状态2.2 索引2.2.1 查看所有索…

Scrum 敏捷开发

什么是敏捷开发 敏捷 开发是一个术语&#xff0c;用于描述迭代软件开发。 迭代软件开发通过在短增量完成工作&#xff08;通常称为 冲刺&#xff0c; Sprint&#xff09;来缩短 DevOps 生命周期。 冲刺通常长达一到四周。 敏捷开发通常与传统或瀑布式开发形成鲜明对比&#xff…

Vue基础9之脚手架的使用、ref属性、props配置项和mixin混入

Vue基础9使用Vue脚手架初始化脚手架说明具体步骤项目文件介绍将前面写好的单文件组件放入这里运行脚手架文件结构render的作用修改默认配置配置项ref属性props配置项简单的传值方法默认的字符串传值使用v-bind对数字类型进行传值限制数据类型接收数据时候只对数据类型进行限制接…

Java 搜索二维矩阵 II

搜索二维矩阵 II中等编写一个高效的算法来搜索 m x n 矩阵 matrix 中的一个目标值 target 。该矩阵具有以下特性&#xff1a;每行的元素从左到右升序排列。每列的元素从上到下升序排列。示例 1&#xff1a;输入&#xff1a;matrix [[1,4,7,11,15],[2,5,8,12,19],[3,6,9,16,22]…

v-model表单

1、v-model的基本使用 表单提交是开发中非常常见的功能&#xff0c;也是和用户交互的重要手段&#xff1a; 比如用户在登录、注册时需要提交账号密码&#xff1b;比如用户在检索、创建、更新信息时&#xff0c;需要提交一些数据&#xff1b; 这些都要求我们可以在代码逻辑中获…