【华为机考】专题突破 第一周:单调栈 739 、503 、901、84

刷题顺序参考于 《2023华为机考刷题指南:八周机考速通车》

前言

单调栈:分为单调递增和单调递减栈。(栈内元素成递增或者递减性):

  • 单调递增栈:从栈底到栈顶数据是从大到小,即 栈内的元素从栈顶 到栈底 是递增的,也就是栈顶的最小,栈底最大;
  • 单调递减栈:从栈底到栈顶数据是从小到大,栈内的元素从栈顶到栈底 是递减的,也就是栈顶的最大,栈底最小。

单调栈的性质:元素加入栈前会把栈顶破坏单调性的元素删除;一般使用单调栈的题目具有以下的两点:离自己最近(栈的后进先出的性质),比自己大(小)、高(低);

单调栈的应用:单调栈一般是用来解决需要 寻找当前点的左右边界最大最小值 之类的问题 ;主要应用:

  • 如果是求右边的第一个最大,那么就是从右向左遍历,构建单调递增栈;
  • 如果是求右边的第一个最小,那么就是从右向左遍历,构建单调递减栈;
  • 如果是求左边的第一个最大,那么就是从左向右遍历,构建单调递增栈;
  • 如果是求左边的第一个最小,那么就是从左向右遍历,构建单调递减栈;

……
简言之:求第一个最大用单调递增栈,求第一个最小用单调递减栈,根据左右,选择是前向遍历还是后向遍历。

单调栈使用模板

stack<Integer> stack = new Stack<>();
for (遍历这个数组){
   if (栈空 || 栈顶元素大于等于当前比较元素){
       入栈;
   }
   else{
       while (栈不为空 && 栈顶元素小于当前元素){
            栈顶元素出栈;
            更新结果;
        }
        当前数据入栈;
   }
}

更多内容可参考:
[1] 单调栈什么时候从后向前遍历,什么时候从前向后遍历?
[2] 刷题笔记6(浅谈单调栈)

一、739. 每日温度

1、题目描述

给定一个整数数组 temperatures ,表示每天的温度,返回一个数组 answer ,其中 answer[i] 是指对于第 i 天,下一个更高温度出现在几天后。如果气温在这之后都不会升高,请在该位置用 0 来代替。

2、测试用例

在这里插入图片描述

3、题解

根据题目分析,要求我们找出右边第一个最大,那么根据前言中的总结,就需要我们从右向左(倒序遍历)的构建单调递增栈。

3.1 单调栈 (⭐)

时间复杂度 O(n) ,空间复杂度 O(n)

[73,74,75,71,69,72,76,73] 为例的单调栈过程:
在这里插入图片描述

class Solution {
    // 思路分析:
    // 1. 非最高温度 对应 下一个更高温度与当前温度的位置差
    // 2. 数组中当前最高 对应 0
    
    // 难点在于:
    // - 如何确定当前数 较之 后面的元素 是否为最大
    
    // Solution1:枚举思路:(遇事不决,暴力枚举!)
    // - 让每个元素都与其后面的元素作比较,直到找到比当前元素大的元素或结束
    // - 时间复杂度为 O(n^2),超出时间限制
    
    // 如何进行优化,降低时间复杂度呢?
    // 枚举的高时间复杂度原因在于 对于每一个元素的最坏情况 都有可能是遍历完整个数组
    // 那么是否有办法很快就让当前元素 确定 下一个比它大的数呢?
    
    // Solution2:单调栈:从后往前遍历,既然从前往后遍历,我们在当前位置就确定谁大谁小(因为没遍历到后面的元素),
    // 那么我们就从后往前遍历,后面的遍历过了,当前的只需要判断就可以

    public int[] dailyTemperatures(int[] temperatures) {
        // 无效输入判断
        int n = temperatures.length;
        if (n <= 0) return new int[0];
        // 定义结果数组,用来返回结果,数组会默认初始化,数组元素全为0
        int[] res = new int[n];
        // 定义一个栈,用来实现单调栈,栈中元素保持 从 栈底到栈顶 数据是从大到小
        Stack<Integer> stack = new Stack<>();
        // 从后向前遍历
        for (int i = n-1; i >= 0; i--) {
            // 如果栈中元素非空,且当前元素大于栈顶元素,则说明后面没有比当天的气温高,弹出栈顶元素
            while (!stack.isEmpty() && temperatures[i] >= temperatures[stack.peek()]) 
                stack.pop();
            // 比较完毕后,如果此时栈为空的话,则说明后面没有比当天更高的温度,该天即为最高温度,赋值为 0
            // 如果栈不为空,则说明后面存在比当天更高的温度,那么让 此时的 栈顶元素 - i 即可
            res[i] = stack.isEmpty() ? 0 : stack.peek() - i;
            // 元素入栈
            stack.push(i);
        }
    
        // 循环结束,返回结果
        return res;
    }  
}

在这里插入图片描述

3.2 暴力枚举 (超时)

时间复杂度 O(n2) ,空间复杂度 O(n)

class Solution {
    // 枚举思路:
    // - 让每个元素都与其后面的元素作比较,直到找到比当前元素大的元素或结束
    public int[] dailyTemperatures(int[] temperatures) {
        // 无效输入判断
        if (temperatures.length <= 0) return new int[0];
        // 定义结果数组,用来返回结果,数组会默认初始化,数组元素全为0
        int[] res = new int[temperatures.length];
        // 双循环遍历
        for (int i = 0; i < temperatures.length-1; i++) {
            for (int j = i+1; j < temperatures.length; j++) {
                if (temperatures[i] < temperatures[j]) {
                    res[i] = j-i;
                    break;
                }
            }
        }
        // 循环结束,返回结果
        return res;
    }  
}

在这里插入图片描述

二、503. 下一个更大元素 II

1、题目描述

给定一个循环数组 nums ( nums[nums.length - 1] 的下一个元素是 nums[0] ),返回 nums 中每个元素的 下一个更大元素 。

数字 x 的 下一个更大的元素 是按数组遍历顺序,这个数字之后的第一个比它更大的数,这意味着你应该循环地搜索它的下一个更大的数。如果不存在,则输出 -1 。

2、测试用例

在这里插入图片描述

3、题解

与上一题 739 不同,该题给定的数组是一个循环数组, 所以要考虑的更多,简单的说就是之前我们只考虑每个元素的后面就行了,现在循环数组的话他前面的元素也有可能成为它的下一个更大元素。

Note:循环数组的下标可以用取余的方式减少重复循环,达到直接定位的效果,如在本题中就通过了 i%n 的方式实现了对数组下标的定位,更多的相关内容可参考 【LeetCode】剑指 Offer 62. 圆圈中最后剩下的数字(约瑟夫环问题) p300 – Java Version(模拟环形链表)。

3.1 两次遍历 + 单调栈 (⭐)

时间复杂度 O(n) ,空间复杂度 O(n)
在这里插入图片描述

Q1:元素右边第一大可能是其左边的元素,那么是否可以通过单调栈找左第一大的方式找到该元素?
不能!与一开始的分析和上图可知,由于形参 nums 是一个循环数组,那么其中的某一元素的右边第一大就有可能是其左边的元素,那么我们该如何来解决这个问题呢?是要实现两个单调栈,一个去找右第一大,一个去找左第一大吗?非也!因为找左第一大,不符合题意,由于是循环数组的原因,1 的右第一大应该是 5,如果我们去找了 1 的左第一大,那么找到的结果是 2,很明显不符合题意。

那么正确的做法就应该是:

  1. 将循环范围扩大到数组长度的 2 倍,通过 索引 % 数组长度的这种求余方式来定位数组下标,这就相当于对数组进行了两次遍历,此时进行的单调栈中就包含了当前元素的左边元素(因为我们在遍历第一遍数组的时候,就已经将数组中最大的元素存入了栈中,所以第二次遍历的时候,如果当前元素它的右边第一大是它的某个左元素,那么这个左元素此时一定存储在了单调栈中)
  2. 直接定义一个二倍数组长度的临时数组,将入参数组 nums 中的元素拷贝两次,然后倒序实现单调递增栈即可,该题解写法可见 3.2
class Solution {
    // Solution1: 两次遍历 + 单调栈
    public int[] nextGreaterElements(int[] nums) {
        
        int n = nums.length; // 获得数组长度
        Stack<Integer> rs = new Stack<>();  // 倒序遍历 找 右第一大
        int[] res = new int[n]; // 定义返回数组,用来存储返回结果

        for (int i = 2*n-1; i >= 0; i--) // 倒序找右第一大,通过取余的方式相当于人工拼接了一次数组
        {
            while (!rs.isEmpty() && nums[i%n] >= nums[rs.peek()]) rs.pop();
            res[i%n] = rs.isEmpty() ? -1 : nums[rs.peek()]; // 定位右第一大元素是谁
            rs.push(i%n); // 将数组元素下标依次压栈
        }

        return res;
    }
}

在这里插入图片描述

3.2 double数组 + 单调栈

时间复杂度 O(n) ,空间复杂度 O(n)

class Solution {
    // Solution2: double数组 + 单调栈
    public int[] nextGreaterElements(int[] nums) {
        
        int n = nums.length; // 获得数组长度
        Stack<Integer> rs = new Stack<>();  // 倒序遍历 找 右第一大
        int[] res = new int[n]; // 定义返回数组,用来存储返回结果

        int[] tmp = new int[2*n]; // 拷贝两次输入数组
        for (int i = 0; i < 2*n; i++) 
        {
            tmp[i] = nums[i%n];
            //System.out.println(tmp[i]);
        }
        
        for (int i = 2*n-1; i >= 0; i--) // 倒序找右第一大
        {
            while (!rs.isEmpty() && tmp[i] >= tmp[rs.peek()]) rs.pop();
            res[i%n] = rs.isEmpty() ? -1 : tmp[rs.peek()]; // 定位右第一大元素是谁
            rs.push(i); // 将数组元素下标依次压栈
        }

        return res;
    }
}

在这里插入图片描述

三、901. 股票价格跨度

1、题目描述

设计一个算法收集某些股票的每日报价,并返回该股票当日价格的 跨度
当日股票价格的 跨度 被定义为股票价格小于或等于今天价格的最大连续日数(从今天开始往回数,包括今天)。

  • 例如,如果未来 7 天股票的价格是 [100,80,60,70,60,75,85],那么股票跨度将是 [1,1,1,2,1,4,6]

在这里插入图片描述

2、测试用例

在这里插入图片描述

3、题解

该题与前两题不同之处在于:前两题都是找右第一大,本题是来找左第一大,所以根据前言中的模板,此时我们需要从左到右的去构建一个 从栈顶到栈底 单调递增栈,通过这种方式,我们就可以从左向右遍历所有元素,如果遍历到的当前元素 大于等于 栈顶元素,则弹出栈顶元素,直至栈中出现了比当前元素大的数,才停止弹出,此时用当前遍历元素的下标 减去 栈顶元素的下标即为 今日股票价格的跨度。

3.1 单调栈 + 二元对 (⭐)

时间复杂度 O(n) ,空间复杂度 O(n)
以 [100,80,60,70,60,75] 为例的出出入栈过程,如下图所示:
在这里插入图片描述

由于本题中没有给定类似数组之类的输入参数,而是需要我们自己去补充完成 next 方法,所以我们没有办法直接获取到输入数据的下标,那么此时就需要我们从 0 开始定义它的下标 idx,并将其同价格一同存储起来,我们可以使用二元数对 int[2] 来存储股票价格的下标(即天数)和股票价格(其中,int[0] 为当前股票价格下标,int[1]为当前股票价格)。同时,我们可以在栈中先插入一个最大值作为天数为 −1 天的价格,来保证栈不会为空。调用 next 时,时,先将栈中价格小于等于此时 price 的元素都弹出,直到遇到一个大于 price 的值,并将 price 入栈,计算下标差后返回。

class StockSpanner {
    Stack<int[]> stack;
    int idx;

    public StockSpanner() {  // 构造函数:用来初始化定义的数据结构
        stack = new Stack<int[]>();  
        stack.push(new int[]{-1, Integer.MAX_VALUE});
        idx = -1;
    }

    public int next(int price) {    
        idx++;
        while (price >= stack.peek()[1]) {  // 二元对:[0]代表下标索引,[1]代表价格
            stack.pop();
        }
        int ret = idx - stack.peek()[0];  // 下标差
        stack.push(new int[]{idx, price});
        return ret;
    }
}
/**
 * Your StockSpanner object will be instantiated and called as such:
 * StockSpanner obj = new StockSpanner();
 * int param_1 = obj.next(price);
 */

在这里插入图片描述

3.2 单调栈 + 哈希表

如果不使用下标索引呢,那么该题就需要在遍历的过程中进行累加操作了,因为我们是从左向右遍历得出每个元素到其左第一大元素之间的跨度,那么当前元素的跨度,就会栈顶元素与它的左第一大元素之间产生了关系(如果当前元素比栈顶元素大,那么也就说明,当前栈顶元素到其左第一大元素之间的跨度属于当前元素中的一部分,然后再比较当前元素与栈顶元素的左第一大元素,如果还是当前元素要大,那么就继续比较下一个左第一大元素,直至找到比当前元素大的元素为止)

当然,即使不使用下标索引,我们还是需要一个二元数对来保存两个元素,一个是当前的股票价格,另一个就是要返回的结果:股票的跨度;那么除了用数组来存储,我们也可以使用哈希表(k-v) 来进行存储,让 key 为股票价格,value 为跨度。

Note:这里还有一个小的注意点,要注意 map.put(price,count) 的位置,不要将其写在循环里面,因为如果入参中存在重复元素时,map.get(stack.peek()) 时就有可能获得到之前存在的,然而新存在的又会覆盖掉之前的,累加起来就会导致结果错误。

class StockSpanner {
    HashMap<Integer,Integer> map;
    Stack<Integer> stack;
    
    public StockSpanner() {
        map=new HashMap<>();
        stack=new Stack<>();
        stack.push(Integer.MAX_VALUE); // 哨兵 永远在栈中 
    }
    
    public int next(int price) {
        int count = 1;
        while(price >= stack.peek()){
            count += map.get(stack.peek());
            stack.pop();
        }
        stack.push(price);
        map.put(price,count);
        return count;
    }
}

在这里插入图片描述

3.3 其它方法(两个栈/数组指针)

两个栈方法可参考:股票价格跨度 – LeetCode
数组+指针方法可参考:【爪哇缪斯】图解LeetCode

四、84. 柱状图中最大的矩形

1、题目描述

给定 n 个非负整数,用来表示柱状图中各个柱子的高度。每个柱子彼此相邻,且宽度为 1 。

求在该柱状图中,能够勾勒出来的矩形的最大面积。

2、测试用例

在这里插入图片描述
在这里插入图片描述

3、题解

笔试面试高频题。该题也可理解成找左边第一小,但找左边第一小只是为了确定左边界,还需经过面积公式的操作求出柱形图中可能的矩阵的最大面积。

3.1 单调栈 (⭐)

时间复杂度 O(n) ,空间复杂度 O(n)
以 [2,1,5,6,2,3] 为例:
在这里插入图片描述

如图所示,当数组遍历到2时(第一张图中的三角处),触发条件,开始计算高度为 6 时的最大矩形面积,以及高度为 5 时的最大矩形面积(此时得到柱形图中矩形的最大面积 10);然后当数组遍历到最后一个元素(我们人为补0的位置),触发条件(弹出栈中所有大于 0 的元素 ==> 相当于清空栈中元素,看看还有哪个高度的矩形面积没有被计算,然后计算一下),开始计算高度为 3/2/1 时的最大矩阵面积,所以这也是为什么要一开始人为改造数组,给数组首尾补 0 的原因,即 保证栈内元素非空,并且对数组中每种情况的高度都能计算一遍

Note:还有需要注意的一点是(width = r - l - 1),这是因为我们对数组进行了补 0 操作,让其左边界定位到了当前元素的左第一小处,所以需要额外减 1,这个地方也可以在给左边界赋值的时候人为加1。

class Solution {
    // Solution1:单调栈

    // 为了数据的正确进行,要改装一下 输入数组,即 为其首尾加0
    // 1. 为什么要在heights最前面加0?
    // 因为没有在heights前加0,不能保证stack不为空,所以left的值就需要赋初始值0
    // 2. 为什么要在heights最后面加0?
    // 在最后加0可以保证矩形高度都是递增的特殊情况下ans也能进行计算
    // 简言之:我们要找每个柱子的左右边界,那么首个柱子缺乏左边界,末尾柱子缺乏右边界

    public int largestRectangleArea(int[] heights) {
        // 无效输入判断
        int len = heights.length;
        if (len <= 0) return -1;
        // 改造数组,为其前后补0
        int[] newHeights = new int[len+2];
        for (int i = 1; i < newHeights.length-1; i++) {
            newHeights[i] = heights[i-1];
        }
        // 构建单调递减栈,找到左右边第一小
        Stack<Integer> stack = new Stack<>();
        // 记录数组中可能的最大面积
        int maxArea = 0;
        // 遍历改造数组
        for (int i = 0; i < newHeights.length; i++) {
            // 栈非空,且当前数组元素小于栈顶元素时,弹出栈顶
            while (!stack.isEmpty() && newHeights[i] < newHeights[stack.peek()]) {
                // 记录当前高度的下标索引(当遇到当前数组元素小于栈顶元素时,即找到了当前高度区域面积的右边界)
                int idx = stack.pop();
                // 左边界
                int l = stack.peek();
                // 右边界
                int r = i;
                // 存储当前高度的最大面积
                maxArea = Math.max(maxArea, newHeights[idx] * (r - l - 1));
            }
            stack.push(i);
        }
        return maxArea;
    }
}

3.2 公式优化(s=(right-left-1)*height[i])

更多内容可参考:柱状图中最大的矩形(难)
根据上面的分析,我们知道对于第i根柱子所围成的最大矩形是 s=(right-left-1)*height[i],其中 right 是右边比它小的柱子的下标,left 是左边比它小的柱子的下标,height[i] 是当前柱子的高度。那么只要我们知道了每根柱子左右两边比它小的值,我们就可以求出最大面积。

我们可以通过以下代码找到当前柱子左边比它小的柱子的下标:

    for (int i = 1; i < height.length; i++) {
        int p = i - 1;
        while (p >= 0 && height[p] >= height[i]) {
            p = leftLess[p];
        }
        leftLess[i] = p;
    }

找右边第一小也同理,于是我们就得到了以下代码:

class Solution {
    public int largestRectangleArea(int[] height) {
        if (height == null || height.length == 0) {
            return 0;
        }
        //存放左边比它小的下标
        int[] leftLess = new int[height.length];
        //存放右边比它小的下标
        int[] rightLess = new int[height.length];
        rightLess[height.length - 1] = height.length;
        leftLess[0] = -1;

        //计算每个柱子左边比它小的柱子的下标
        for (int i = 1; i < height.length; i++) {
            int p = i - 1;
            while (p >= 0 && height[p] >= height[i]) {
                p = leftLess[p];
            }
            leftLess[i] = p;
        }
        //计算每个柱子右边比它小的柱子的下标
        for (int i = height.length - 2; i >= 0; i--) {
            int p = i + 1;
            while (p < height.length && height[p] >= height[i]) {
                p = rightLess[p];
            }
            rightLess[i] = p;
        }
        int maxArea = 0;
        //以每个柱子的高度为矩形的高,计算矩形的面积。
        for (int i = 0; i < height.length; i++) {
            maxArea = Math.max(maxArea, height[i] * (rightLess[i] - leftLess[i] - 1));
        }
        return maxArea;
    }
}

在这里插入图片描述

五、小结

专题突破 第一周:单调栈 739 、503 、901、84, 一共分为四小题:

  • 题目一:739. 每日温度,Medium,找右第一大
  • 题目二:503. 下一个更大元素 II,Medium,同样也是找右第一大,但不同之处在于输入参数为循环数组,需要两次遍历并对下标进行求余计算;
  • 题目三:901. 股票价格跨度,Medium,找左第一大,但由于该题没有给定输入参数(eg:数组等存储元素的数据结构),所以此处需要我们自己定义一个二元数对(int[] 或 哈希表)来存储下标 或 结果;
  • 题目四:84. 柱状图中最大的矩形,Hard,1. 找左右第一小,该题由于要计算矩形的面积,那么就需要我们去找到当前柱子的左右第一小柱子的下标来作为左右边界(宽度),我们可以从左向右遍历,去找左第一小,如果当前元素小于栈顶元素,则弹出栈顶元素,直至栈顶元素小于或等于当前元素,那么当前元素的下标即为右边界,当前栈顶元素的下标即为左边界,被弹出的栈顶元素对应数组的值即为高(单调栈做法);2. 除了单调栈的做法外,我们也可以直接从当前元素位置循环去找当前元素的左右第一小,然后用公式计算得出最大矩形(效率更高)。

六、参考资料

[1] 单调栈+hashmap 简单易理解的解法 – 901
[2] 股票价格跨度 – 901
[3] 动画演示 单调栈 84.柱状图中最大的矩形 – 84 – 编程狂想曲

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【手把手刷CCF】202303-2-垦田计划100分(超简单思路,含详细解释注释与代码)

文章目录&#xff1a; 故事的开头总是极尽温柔&#xff0c;故事会一直温柔……&#x1f49c;一、&#x1f333;代码如下&#xff1a;二、&#x1f335;解题思路❤️❤️❤️忙碌的敲代码也不要忘了浪漫鸭&#xff01; 故事的开头总是极尽温柔&#xff0c;故事会一直温柔……&am…

SpringBoot集成WebSocket实现及时通讯聊天功能!!!

1&#xff1a;在SpringBoot的pom.xml文件里添加依赖: <!-- websocket --> <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-websocket</artifactId> </dependency> 2&#xff1a;在配置中…

java 拼接字符串的方法

1.拼接字符串的方法&#xff0c;先要将字符串转化为数字类型&#xff0c;再根据需要拼接。这样可以避免直接拼接导致的错误。 2.将字符串转化为数字类型&#xff0c;这个就是一个循环。可以使用循环的方法&#xff0c;但是循环次数不宜太多&#xff0c;否则容易出错。 3.可以使…

ERTEC200P-2 PROFINET设备完全开发手册(7-1)

7. 配置模块及自定义模块 7.1.1 PN设备的基本模型 初次接触PN的开发者&#xff0c;最容易出现的错误就是设备的实际配置与TIA的组态不一致。为了开发的过程更加顺利&#xff0c;非常有必要掌握PN设备的基础模型。PN设备的基本模型如下图描述&#xff1a; PN设备的基本构成是插…

论文的总体结构及质量控制

要写出一篇高质量AI领域的论文&#xff0c;首先要搞清楚论文由哪几部分组成&#xff0c;即论文的总体结构。同时&#xff0c;还要了解AI论文的质量评价与质量控制的指标。这样做的目的是为了弄明白AI论文的结构以及什么样的AI论文才是好的论文。 通常一篇AI论文的总体结构主…

ZVL3网络分析仪

ZVL3 Rohde&Schwarz ZVL3 3G矢量网络分析仪|罗德与施瓦茨 9KHz至3GHz 罗德与施瓦茨Rohde&Schwarz 性能特点&#xff1a; 频率范围 9kHz至3GHz/6 GHz(典型值为5kHz) 测量时间(201个测量点&#xff0c;以校准的双端口) <75ms 数据传输(201个测量点) 在100Mbit/sLAN…

DFS与BFS|树与图的遍历:拓扑排序

深度优先搜索DFS DFS每次往最深处搜&#xff0c;搜到叶子节点就返回&#xff0c;然后继续搜&#xff0c;特点&#xff1a;走到头才返回&#xff0c;返回并不是返回最开始&#xff0c;而是每次返回上一层之后&#xff0c;再看这一层能不能往下搜 DFS有回溯和剪枝。返回上一层的过…

OpenCV实例(七)汽车检测

OpenCV实例&#xff08;七&#xff09;汽车检测 1.概述2.代码实例3.代码功能 作者&#xff1a;Xiou 1.概述 对于图像和视频检测中的目标类型并没有具体限制&#xff0c;但是&#xff0c;为了使结果的准确度在可接受范围内&#xff0c;需要一个足够大的数据集&#xff0c;包括…

《Spring MVC》 第四章 域对象、视图、转发和重定向

前言 介绍Spring MVC的域对象、视图、转发和重定向 1、域对象共享数据 Spring MVC 提供了多种域对象共享数据的方式&#xff0c;其中最常用的方式如下&#xff1a; 1.1、使用 Servlet API 向 request 域对象中共享数据 服务端代码&#xff1a; RequestMapping("toLo…

为什么企业都需要搭建搭建一个内部知识库?

企业内部知识管理是指企业通过各种手段收集、整理、管理和传播企业内部的知识&#xff0c;以提高企业的竞争力和创新能力。在实践中&#xff0c;企业内部知识管理往往需要建立一个内部知识库&#xff0c;以更好地实现知识的共享和管理。本文将从以下几个方面探讨为什么企业内部…

【SWAT水文模型】ArcSWAT输入准备

ArcSWAT输入准备 1 必需的ArcSWAT空间数据集1.1 数字高程模型&#xff08;DEM&#xff09;1.2 土地覆盖/土地利用类型1.3 土壤数据 2 可选的ArcSWAT空间数据集2.1 DEM Mask2.2 Streams2.3 User- Defined Watersheds 3 ArcSWAT表格和文本文件3.1 子流域出口位置表(dBase 表)3.2 …

掏空腰包,日子难过,机缘转岗软件测试,这100个日夜的心酸只有自己知道...

我今年27岁&#xff0c;原本从事着土木工程相关的工作&#xff0c;19年开始有了转行的想法... 大学刚毕业那年&#xff0c;我由于学的是土木工程专业&#xff0c;自然而然的从事了和土木工程相关的工作&#xff0c;房贷、车贷&#xff0c;在经济的高压下&#xff0c;当代社会许…

Idea 配置 maven 离线使用

首先&#xff0c;项目中的依赖已经下载到本地仓库&#xff0c;在没有网络或者没办法连通公司的maven仓库时&#xff0c;需要配置离线使用。 1. 配置 setting.xml 在 maven 使用的 setting.xml 文件中&#xff0c;加入以下配置。 默认在 maven安装目录下的 conf 文件夹下 。 &…

没看错!一行python代码就可以帮您获取图片中的文字信息

最近工作中有需求需要用python对图片中的文字进行识别&#xff0c;调研了一下&#xff0c;选择了tesseract&#xff0c; 目前在github上有50.5k个star&#xff01;python可以调用&#xff0c;安装也十分方便&#xff0c;pip install pytesseract 即可。如果没有Pillow 包&…

关于yolov8的一些理解

文章目录 1.前言2.创新点及工作3. 网络结构3.1 BackBone3.1.1 C2F3.1.2 结构微调3.1.2 SPPF 3.2 Neck3.3 Head 4.正样本匹配策略4.1 静态分配策略&动态分配策略4.2 TaskAlignedAssigner 5.损失函数5.1 概述5.2 Distribution Focal Loss 6.总结 1.前言 YOLOv8 是 ultralyti…

vCener 配置 vSan 网络

文章目录 1. 准备2. 创建vsan网络2.1 创建 vSphere Distributed Switch &#xff08;vds&#xff09;2.2 添加管理主机2.3 添加 networking 3. 删除3.1 删除 vmkernel adapter3.2 删除 hosts3.3 删除 DSwitch 1. 准备 三台物理机搭建 exsi一台部署 vcenter 管理三台 exsi每台物…

如何计算连续变量的熵

背景 做特征选择时&#xff0c;有时候会用到计算特征的信息熵&#xff0c;可是离散的好计算&#xff0c;但连续的呢&#xff1f;按照把连续变量离散的方法设置阈值点吗&#xff1f;好像比较麻烦&#xff0c;需要排序&#xff0c; 计算阈值。没有能自动的方法吗&#xff1f; 找…

Linux入门 - 最常用基础指令汇总

目录 ls指令 pwd指令 cd指令 touch指令 mkdir指令 rmdir指令 && rm 指令 man指令&#xff08;重要&#xff09; cp指令&#xff08;重要&#xff09; mv指令&#xff08;重要&#xff09; cat指令 more指令 less指令&#xff08;重要&#xff09; head指令…

3.4 迭代法

4.1 雅克比迭代法&#xff1a; 雅可比迭代法是一种用于求解线性方程组的迭代算法&#xff0c;其基本思想是将线性方程组中的系数矩阵拆分为对角线矩阵和非对角线矩阵两部分&#xff0c;并利用对角线矩阵的逆矩阵来迭代求解方程组。 具体地&#xff0c;设线性方程组为Axb&…