算法设计与分析实验:滑动窗口与二分查找

news2025/1/23 11:26:10

目录

一、寻找两个正序数组的中位数

1.1 具体思路

1.2 流程展示

1.3 代码实现

1.4 代码复杂度分析

1.5 运行结果

二、X的平方根

2.1 具体思路

2.2 流程展示

2.3 代码实现

2.4 代码复杂度分析

2.5 运行结果

三、两数之和 II-输入有序数组

3.1 采用二分查找的思想

3.2 采用滑动窗口的思想

四、二叉搜索树中的插入操作

4.1 采用递归的思想

4.2 采用迭代的思想


一、寻找两个正序数组的中位数

力扣第四题

1.1 具体思路

本题采用二分查找的思想。

(1)按照题目中的条件,首先确保 nums1 中的长度小于等于 nums2 中的长度。如果不是这样,则交换数组以确保这个条件。

(2)初始化两个指针,在 nums1 中的切分位置 left 从 0 开始、right 从 nums1 的长度开始。在 nums1 中选取一个中间的位置进行切分,即 partition1 = (left + right) // 2。同时,根据 partition1 在 nums2 中对应的切分位置,即 partition2 = (m + n + 1) // 2 - partition1。

(3)创造四个变量:max_left1、min_right1、max_left2 和 min_right2。这些变量分别表示 partition1 左边部分的最大值、partition1 右边部分的最小值、partition2 左边部分的最大值和 partition2 右边部分的最小值。有一点需要注意,当 partition1 或 partition2 等于 0 或 m 或 n 的时候,需要将对应的变量赋值为正负无穷来避免越界错误。

(4)进行比较得出结果。如果 max_left1 小于等于 min_right2 且 max_left2 小于等于 min_right1,则找到了合适的切分位置。此时,中位数为 (max(max_left1, max_left2) + min(min_right1, min_right2)) / 2。否则,需要调整切分位置的范围。

(5)不断地调整切分位置的范围,直到找到合适的切分位置来得到中位数。

1.2 流程展示

假设两个数组如下,根据二分思想对其进行初步划分

nums1: [1, 3, 8, 9]          max_left1 = 3      min_right1 = 8

nums2: [2, 7, 10, 12, 15]    max_left2 = 7      min_right2 = 10

             <--- partition1 --->

nums1: 1  |  3  8  9

          x

             <--- partition2 --->

nums2: 2  7  |  10  12  15

             x

在切分位置上,nums1 的左半部分为 [1],右半部分为 [3, 8, 9]

在切分位置上,nums2 的左半部分为 [2, 7],右半部分为 [10, 12, 15]

max_left1 = 1      min_right1 = 3

max_left2 = 7      min_right2 = 10

因为 max_left1 <= min_right2 且 max_left2 > min_right1,所以需要继续调整切分位置。

将切分位置向右移动一位:

             <--- partition1 --->

nums1: 1  3  |  8  9

             x

             <--- partition2 --->

nums2: 2  7  |  10  12  15

             x

在切分位置上,nums1 的左半部分为 [1, 3],右半部分为 [8, 9]

在切分位置上,nums2 的左半部分为 [2, 7],右半部分为 [10, 12, 15]

max_left1 = 3      min_right1 = 8

max_left2 = 7      min_right2 = 10

因为 max_left1 <= min_right2 且 max_left2 <= min_right1,所以找到了合适的切分位置。

中位数:(max(max_left1, max_left2) + min(min_right1, min_right2)) / 2 = (8 + 8) / 2 = 8

1.3 代码实现

from typing import List

class Solution:
    def findMedianSortedArrays(self, nums1: List[int], nums2: List[int]) -> float:
        # 确保 nums1 的长度小于等于 nums2 的长度
        if len(nums1) > len(nums2):
            nums1, nums2 = nums2, nums1

        m, n = len(nums1), len(nums2)
        left, right = 0, m
        median1, median2 = 0, 0

        while left <= right:
            # 在 nums1 的切分位置
            partition1 = (left + right) // 2
            # 在 nums2 中的相应切分位置
            partition2 = (m + n + 1) // 2 - partition1

            # 处理边界情况
            max_left1 = float('-inf') if partition1 == 0 else nums1[partition1 - 1]
            min_right1 = float('inf') if partition1 == m else nums1[partition1]
            max_left2 = float('-inf') if partition2 == 0 else nums2[partition2 - 1]
            min_right2 = float('inf') if partition2 == n else nums2[partition2]

            # 根据切分位置进行比较得出结果
            if max_left1 <= min_right2 and max_left2 <= min_right1:
                median1 = max(max_left1, max_left2)
                median2 = min(min_right1, min_right2)
                break
            elif max_left1 > min_right2:
                right = partition1 - 1
            else:
                left = partition1 + 1

        # 根据数组的总长度是奇数还是偶数来返回中位数
        if (m + n) % 2 == 0:
            return (median1 + median2) / 2
        else:
            return median1

nums1 = [1, 3]
nums2 = [2]
solution = Solution()
result = solution.findMedianSortedArrays(nums1, nums2)
print(f"{result:.5f}")  # 输出:2.00000

nums1 = [1, 2]
nums2 = [3, 4]
solution = Solution()
result = solution.findMedianSortedArrays(nums1, nums2)
print(f"{result:.5f}")  # 输出:2.50000

1.4 代码复杂度分析

这个算法的时间复杂度为O(log(min(m, n))),其中m和n分别是nums1和nums2的长度。

算法的核心是使用二分查找来找到一个合适的切分位置,使得切分后的两个部分满足以下条件:

nums1和nums2被切分成左右两部分,分别为nums1_left和nums1_right、nums2_left和nums2_right。

左半部分的元素都小于或等于右半部分的元素。

在每一轮二分查找中,根据切分位置,在nums1和nums2中找到对应的切分点,然后比较切分点的左右元素,根据比较结果调整切分位置的范围。

最坏情况下,算法会进行O(log(min(m, n)))轮二分查找,每一轮查找的时间复杂度为O(1),因此总的时间复杂度为O(log(min(m, n)))。

空间复杂度为O(1),因为算法只使用了有限的额外变量来存储中间结果。没有使用额外的数据结构。

总结起来,该算法的时间复杂度为O(log(min(m, n))),空间复杂度为O(1)。

1.5 运行结果

示例一:nums1 = [13]     nums2 = [2]

示例二:nums1 = [12]     nums2 = [34]

结果如下

输出与预期一致

二、X的平方根


力扣第69题

2.1 具体思路

本题运用二分查找的思想。

首先,算术平方根的范围一定是在 [0, x] 的范围内,因此我们可以对这个范围进行二分查找。

我们用 left 和 right 分别表示左右边界,mid 表示中间位置。将 left 初始化为 0,right 初始化为 x。

每次循环时,让 mid 等于 left 和 right 的平均值。如果 mid 的平方等于 x,则 mid 就是 x 的算术平方根,直接返回 mid。

如果 mid 的平方小于 x,则说明要继续向右搜索算术平方根,在右半部分进行二分查找,因此将 left 更新为 mid+1。

如果 mid 的平方大于 x,则说明要往左边搜索算术平方根,在左半部分进行二分查找,因此将 right 更新为 mid-1。

上述过程重复执行,直到 left > right 为止。此时,left-1 就是 x 的算术平方根的值。

2.2 流程展示

        left           right

         ↓             ↓

  1. ----- ... ------[mid]------ ... ------[x]

   mid^2 < x          mid^2 > x

(left 更新为 mid+1)     (right 更新为 mid-1)

初始时,left 的值为 0,right 的值为 x。

在每次循环中,我们计算 mid 的值,并比较 mid 的平方与 x 的大小关系。并执行如上更新。

这样不断循环直到 left > right 为止,最后返回 left-1,即为 x 的算术平方根的整数部分。

2.3 代码实现

def mySqrt(x: int) -> int:
    left, right = 0, x
    while left <= right:
        mid = (left + right) // 2
        if mid * mid == x:
            return mid
        elif mid * mid < x:
            left = mid + 1
        else:
            right = mid - 1
    return left - 1

# 示例测试代码
print(mySqrt(4))  # 输出:2
print(mySqrt(8))  # 输出:2
print(mySqrt(16))  # 输出:4

2.4 代码复杂度分析

时间复杂度:

while 循环中,每次二分法排除了一半的区间,因此时间复杂度为 O(logn)。

空间复杂度:

只定义了常数个变量,因此空间复杂度为 O(1)。

因此,整体时间复杂度为 O(logn),空间复杂度为 O(1)。

2.5 运行结果

示例有4、8、10

由于返回类型为整数,因此预期的返回结果为2,2,3

输出与预期结果一致

三、两数之和 II-输入有序数组

力扣第167题

3.1 采用二分查找的思想

(1)具体思路

若使用二分查找的思路,我们可以遍历数组中的每个元素,然后针对当前元素,在剩余部分中使用二分查找来寻找与之匹配的目标数。具体步骤如下:

遍历数组中的每个元素 numbers[i]。

在 numbers[i+1:] 的范围内使用二分查找,查找目标数 target - numbers[i]。

如果找到目标数,则返回 [i+1, j+1],其中 j 是目标数在数组中的下标。

如果未找到目标数,则继续遍历下一个元素。若最后仍未找到,则返回[-1,-1]表示未找到。

对于二分查找,我们定义了一个 binary_search 函数,用于实现二分查找。它接收一个有序的整数数组 numbers,目标值 target,以及搜索范围的起始和结束下标 start 和 end。在 binary_search 函数中,我们通过不断更新起始和结束下标来缩小搜索范围,直到找到目标值或者确定目标值不存在。

(2)代码实现

from typing import List


def binary_search(numbers: List[int], target: int, start: int, end: int) -> int:
    while start <= end:
        mid = (start + end) // 2
        if numbers[mid] == target:
            return mid
        elif numbers[mid] < target:
            start = mid + 1
        else:
            end = mid - 1
    return -1


def twoSum(numbers: List[int], target: int) -> List[int]:
    n = len(numbers)
    for i in range(n):
        remaining = target - numbers[i]
        # 在 numbers[i+1:] 中使用二分查找
        j = binary_search(numbers, remaining, i + 1, n - 1)
        if j != -1:
            return [i + 1, j + 1]
    return [-1, -1]


# 示例测试代码
print(twoSum([2, 7, 11, 15], 9))  # 输出:[1, 2]
print(twoSum([2, 3, 4], 6))  # 输出:[1, 3]
print(twoSum([-1, 0], -1))  # 输出:[1, 2]

(3)复杂度分析

这段代码使用了二分查找来实现 twoSum 函数,下面是对其复杂度进行分析:

twoSum 函数中的循环遍历数组,时间复杂度为 O(n),其中 n 是数组的长度。

在每次循环中,调用了 binary_search 函数进行二分查找。二分查找的时间复杂度为 O(logn)。

综上所述,整个算法的时间复杂度为 O(nlogn)。

空间复杂度方面,除了存储结果的列表外,算法没有使用额外的空间,因此空间复杂度为 O(1)。

需要注意的是,输入的数组在调用 twoSum 函数之前没有要求是有序的。如果输入的数组是有序的,则可以使用双指针的方法来进一步优化时间复杂度为 O(n),即我们采用的第二种思想-滑动窗口。

3.2 采用滑动窗口的思想

(1)具体思路

算法思路如下:

定义两个指针 left 和 right,分别指向数组的最左端和最右端。

循环进行以下操作,直到找到满足条件的两个数或者 left >= right:

计算当前窗口的和 currSum = numbers[left] + numbers[right]。

如果 currSum 等于目标数 target,则返回结果 [left + 1, right + 1]。

如果 currSum 大于目标数 target,说明当前窗口的和太大了,需要将窗口右侧向左移动,即 right -= 1。

如果 currSum 小于目标数 target,说明当前窗口的和太小了,需要将窗口左侧向右移动,即 left += 1。

如果循环结束都没有找到满足条件的两个数,则返回 [-1, -1]。

(2)流程展示

假设目标数为9,数组为[2, 4, 6, 8, 10, 12]。

初始情况下,左右指针分别指向数组的最左端和最右端,构成一个长度为数组长度的窗口。窗口内的数之和为14,大于目标数9。

接着,我们将右指针左移一格:

  

当前窗口的数之和为12,仍然大于目标数9。我们再次将右指针左移一格:

  

  

当前窗口的数之和为10,仍然大于目标数9。我们再次将右指针左移一格:

 

当前窗口的数之和为6,小于目标数9。我们将左指针右移一格:

当前窗口的数之和为6,仍然小于目标数9。我们再次将左指针右移一格:

当前窗口的数之和为2,仍然小于目标数9。我们再次将左指针右移一格:  

当前窗口的数之和为0,仍然小于目标数9。此时,左右指针已经相遇,并且没有找到符合要求的两个数。因此返回 [-1, -1] 表示未找到。

(3)代码实现

#滑动窗口
from typing import List

def twoSum(numbers: List[int], target: int) -> List[int]:
    left, right = 0, len(numbers) - 1
    while left < right:
        currSum = numbers[left] + numbers[right]
        if currSum == target:
            return [left + 1, right + 1]
        elif currSum > target:
            right -= 1
        else:
            left += 1
    return [-1, -1]


print(twoSum([2, 7, 11, 15], 9))  # 输出:[1, 2]
print(twoSum([2, 3, 4], 6))  # 输出:[1, 3]
print(twoSum([-1, 0], -1))  # 输出:[1, 2]

(4)代码复杂度分析

这个算法的时间复杂度是 O(n),其中 n是数组的长度。因为左右指针移动的次数最多为 n次,每次移动指针需要 O(1) 的时间,所以总时间复杂度为 O(n)。

空间复杂度为 O(1),因为没有使用额外的数据结构。

(5)运行结果

示例1:[271115], 9

示例2:[234], 6

示例3:[-10], -1

运行结果均符合预计

四、二叉搜索树中的插入操作

力扣第702题

4.1 采用递归的思想

(1)具体思路

如果当前节点为空,则将新值插入到当前位置。

如果新值小于当前节点的值,则继续在当前节点的左子树中插入新值。

如果新值大于当前节点的值,则继续在当前节点的右子树中插入新值。

最后返回根节点。

(2)流程展示

初始树:

          4

        /   \

       2     7

      / \  

     1   3   

插入新值 5:

          4

        /   \

       2     7

      / \     \

     1   3     5

这个过程中,我们首先从根节点开始,比较新值 5 和当前节点值 4,由于 5 大于 4,所以我们移动到右子树继续比较。

在右子树中,新值 5 比当前节点值 7 小,所以我们将新值插入到右子树的左子树位置。

最终的结果是,树中正确地插入了新值 5。

(3)代码实现

class TreeNode:
    def __init__(self, val=0, left=None, right=None):
        self.val = val
        self.left = left
        self.right = right

def insertIntoBST(root: TreeNode, val: int) -> TreeNode:
    if not root:
        return TreeNode(val)
    if val < root.val:
        root.left = insertIntoBST(root.left, val)
    else:
        root.right = insertIntoBST(root.right, val)
    return root

def treeToList(root):
    result = []

    def inorderTraversal(node):
        if not node:
            return
        inorderTraversal(node.left)
        result.append(node.val)
        inorderTraversal(node.right)

    inorderTraversal(root)
    return result

# 创建示例树 [4,2,7,1,3]
root = TreeNode(4)
root.left = TreeNode(2)
root.right = TreeNode(7)
root.left.left = TreeNode(1)
root.left.right = TreeNode(3)

# 插入新值 5
new_val = 5
new_root = insertIntoBST(root, new_val)

# 将树转换为列表
result_list = treeToList(new_root)
print(result_list)

(4)代码复杂度分析

对于插入操作的时间复杂度分析:

最好情况下,树是平衡的,每个节点的左右子树大小相差不大于 1。在这种情况下,插入节点的时间复杂度为 O(logN),其中 N 是树中节点的数量。

最坏情况下,树是非平衡的,形成了一个链表结构。在这种情况下,插入节点的时间复杂度为 O(N),其中 N 是树中节点的数量。

因此,在平均情况下,插入节点的时间复杂度为 O(logN)。

对于将二叉搜索树转换为列表的操作,需要进行中序遍历,即遍历所有节点。因此,时间复杂度是 O(N),其中 N 是树中节点的数量。

综上所述,插入节点的时间复杂度为 O(logN),将树转换为列表的时间复杂度为 O(N)。

空间复杂度方面,由于使用了递归操作,在递归过程中会使用额外的堆栈空间。递归的深度取决于树的高度,在最坏情况下为 O(N),其中 N 是树中节点的数量。因此,空间复杂度为 O(N),除了输入和输出所占用的空间。

4.2 采用迭代的思想

(1)具体思想

如果根节点为空,则直接创建一个新节点作为根节点,返回根节点。

如果插入的值小于当前节点的值,并且当前节点的左子树为空,则将新节点插入为当前节点的左子节点,返回根节点。

如果插入的值大于当前节点的值,并且当前节点的右子树为空,则将新节点插入为当前节点的右子节点,返回根节点。

如果插入的值小于当前节点的值,将当前节点移动到左子节点继续插入的位置。

如果插入的值大于当前节点的值,将当前节点移动到右子节点继续插入的位置。

重复上述步骤直到找到合适的插入位置。

(2)流程展示

首先,创建一个空的二叉搜索树:

          (empty)

接下来,按照迭代思想插入节点的过程:

插入节点值为 4:

             4

插入节点值为 2:

             4

            /

           2

插入节点值为 7:

            4

            / \

           2  7

插入节点值为 1:

            4

            / \

           2  7

          /

         1

插入节点值为 3:

、          4

            / \

           2  7

          / \

         1  3

以上就是按照迭代思想依次插入节点到二叉搜索树中的过程,并得到了最终的二叉搜索树。

(3)代码实现

class TreeNode:
    def __init__(self, val=0, left=None, right=None):
        self.val = val
        self.left = left
        self.right = right

def insertIntoBST(root: TreeNode, val: int) -> TreeNode:
    if not root:
        return TreeNode(val)

    node = root
    while node:
        if val < node.val:
            if not node.left:
                node.left = TreeNode(val)
                break
            else:
                node = node.left
        else:
            if not node.right:
                node.right = TreeNode(val)
                break
            else:
                node = node.right

    return root

# 创建示例树 [4,2,7,1,3]
root = TreeNode(4)
root.left = TreeNode(2)
root.right = TreeNode(7)
root.left.left = TreeNode(1)
root.left.right = TreeNode(3)

# 插入新值 5
new_val = 5
new_root = insertIntoBST(root, new_val)

# 中序遍历树,将树转换为列表
def inorderTraversal(node):
    if not node:
        return []
    return inorderTraversal(node.left) + [node.val] + inorderTraversal(node.right)

result_list = inorderTraversal(new_root)
print(result_list)

(4)复杂度分析
对于插入一个新节点,上述算法的时间复杂度为 O(h),其中 h 为树的高度。

由于二叉搜索树的高度与其中的节点数量相关,因此在最坏情况下(即插入的节点值是原来树中最大或最小的),时间复杂度为 O(n),其中 n 表示树中节点的数量。

但是平衡二叉搜索树(如 AVL 树、红黑树)可以保证树的高度为 O(log n),因此插入新节点的时间复杂度也可以控制在 O(log n) 级别。

对于空间复杂度,递归实现需要调用栈空间,而迭代实现只需要一个额外指针变量 node,因此空间复杂度都是 O(1)。

(5)运行结果

输入:root = [4,2,7,1,3], val = 5

输出:[4,2,7,1,3,5]

记得天天开心哦~~~

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

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

相关文章

【C深度解剖】const关键字

简介&#xff1a;本系列博客为C深度解剖系列内容&#xff0c;以某个点为中心进行相关详细拓展 适宜人群&#xff1a;已大体了解C语法同学 作者留言&#xff1a;本博客相关内容如需转载请注明出处&#xff0c;本人学疏才浅&#xff0c;难免存在些许错误&#xff0c;望留言指正 作…

4核16G10M幻兽帕鲁服务器只需26元/月?!阿里云挑战全网最低价

1月29日&#xff0c;阿里云又发大招&#xff01;将原先4核16G3M的套餐直接升级至10M&#xff0c;价格从32元直降至26元&#xff01; 目前已是全网幻兽帕鲁4-8人游玩配置的服务器性价比极高的套餐&#xff01; 》》阿里云4核16G10M 26元/月 整理至1月29日&#xff0c;各家幻兽…

Ubuntu本地部署Nextcloud并结合内网穿透实现远程访问搭建个人云盘

&#x1f308;个人主页: Aileen_0v0 &#x1f525;热门专栏: 华为鸿蒙系统学习|计算机网络|数据结构与算法 ​&#x1f4ab;个人格言:“没有罗马,那就自己创造罗马~” 文章目录 摘要1. 环境搭建2. 测试局域网访问3. 内网穿透3.1 ubuntu本地安装cpolar3.2 创建隧道3.3 测试公网访…

Mac+Android Studio配置 Flutter环境

Fluttrer中文下载官网 Flutter下载官网 1、环境变量 .zshrc #Flutter export PUB_HOSTED_URL"https://pub.flutter-io.cn" export FLUTTER_STORAGE_BASE_URL"https://storage.flutter-io.cn" export FLUTTER_HOME/Users/leon/Flutter/flutter_3_10_4/f…

强化学习-google football 实验记录

google football 实验记录 1. gru模型和dense模型对比实验 实验场景&#xff1a;5v5(控制蓝方一名激活球员)&#xff0c;跳4帧&#xff0c;即每个动作执行4次 实验点&#xff1a; 修复dense奖励后智能体训练效果能否符合预期 实验目的&#xff1a; 对比gru 长度为16 和 dens…

【前端web入门第二天】02 表单-input标签-单选框-多选框

表单 文章目录: 1.input标签基本使用 1.1 input标签占位文本1.2 单选框 radio 1.3 多选框 checkbox 作用:收集用户信息。 使用场景: 登录页面注册页面搜索区域 1.input标签基本使用 input标签type属性值不同&#xff0c;则功能不同。 <input type"..."&g…

如何搭建开源笔记Joplin服务并实现远程访问本地数据

文章目录 1. 安装Docker2. 自建Joplin服务器3. 搭建Joplin Sever4. 安装cpolar内网穿透5. 创建远程连接的固定公网地址 Joplin 是一个开源的笔记工具&#xff0c;拥有 Windows/macOS/Linux/iOS/Android/Terminal 版本的客户端。多端同步功能是笔记工具最重要的功能&#xff0c;…

【深度学习:目标检测】深度学习中目标检测模型、用例和示例

【深度学习&#xff1a;目标检测】深度学习中目标检测模型、用例和示例 什么是物体检测&#xff1f;物体检测与图像分类物体检测与图像分割 计算机视觉中的目标检测物体检测的优点物体检测的缺点深度学习和目标检测人员检测 物体检测如何工作&#xff1f;一阶段与两阶段深度学习…

从0开始搭建若依微服务项目 RuoYi-Cloud(保姆式教程 一)

掌握陌生项目解读技巧 掌握若依(RuoYi-Cloud)框架 掌握SpringCloud Alibaba体系项目开发套路&#xff0c;结合我之前所有企业项目来学习就知道有多么简单。 一、框架介绍 1. 简介 一直想做一款后台管理系统&#xff0c;看了很多优秀的开源项目但是发现没有合适的。于是利用空…

老司机用脚本批量巧删恶意文件

作者&#xff1a;田逸&#xff08;formyz&#xff09; 一个NFS服务器&#xff0c;为多个Web项目所共享。这些目录包括PHP程序、图片、HTML页面和用户上传的文档和附件等。因为某些Web框架古老&#xff0c;存在诸如不对上传文件做严格的安全性检查&#xff0c;虽然此NFS服务器位…

OceanMind海睿思入选《2023大数据产业年度创新技术突破奖》,并蝉联多项图谱

近日&#xff0c;由数据猿和上海大数据联盟主办&#xff0c;上海市经济和信息化委员会、上海市科学技术委员会指导的“第六届金猿季&魔方论坛——大数据产业发展论坛”在上海成功举行&#xff0c;吸引了数百位业界精英的参与。中新赛克海睿思作为国内数字化转型优秀厂商代表…

虚拟机安装Centos8.5

记得看目录哦&#xff01; 附件1. 新建虚拟机2. 安装Centos8.5 附件 安装包自行下载 https://mirrors.aliyun.com/centos/8/isos/x86_64/ 1. 新建虚拟机 2. 安装Centos8.5 启动虚拟机–选择第一个install Centos8.5 记得接收许可证

25考研北大软微该怎么做?

25考研想准备北大软微&#xff0c;那肯定要认真准备了 考软微需要多少实力 现在的软微已经不是以前的软微了&#xff0c;基本上所有考计算机的同学都知道&#xff0c;已经没有什么信息优势了&#xff0c;只有实打实的有实力的选手才建议报考。 因为软微的专业课也是11408&am…

在Windows上安装与配置Apache服务并结合内网穿透工具实现公网远程访问本地内网服务

文章目录 前言1.Apache服务安装配置1.1 进入官网下载安装包1.2 Apache服务配置 2.安装cpolar内网穿透2.1 注册cpolar账号2.2 下载cpolar客户端 3. 获取远程桌面公网地址3.1 登录cpolar web ui管理界面3.2 创建公网地址 4. 固定公网地址 前言 Apache作为全球使用较高的Web服务器…

Web开发8:前后端分离开发

在现代的 Web 开发中&#xff0c;前后端分离开发已经成为了一种常见的架构模式。它的优势在于前端和后端可以独立开发&#xff0c;互不干扰&#xff0c;同时也提供了更好的可扩展性和灵活性。本篇博客将介绍前后端分离开发的概念、优势以及如何实现。 什么是前后端分离开发&am…

Linux-动静态库

背景 在实践中&#xff0c;我们一定会使用别人的库&#xff08;不限于C、C的库&#xff09;&#xff0c;在实践中&#xff0c;我们会使用成熟、被广泛使用的第三方库&#xff0c;而不会花费很多时间自己造轮子&#xff0c;为了能更好地使用库&#xff0c;就要在学习阶段了解其…

[嵌入式系统-4]:龙芯1B 开发学习套件-1-开发版硬件介绍

目录 前言&#xff1a; 一、龙芯 1B 开发学习套件简介 1.1 概述 二、龙芯1B 200开发板硬件组成与接口介绍 2.1 概述 2.2 核心板 2.2.1 CPU 2.2.2 什么是核心板 2.2.3 龙芯1B 200核心板 2.2.4 龙芯1B核心板的接口定义 2.3 开发板 2.3.1 龙芯1B0200开发板 2.3.2 龙芯…

Linux使用匿名管道实现进程池得以高效通信

&#x1f3ac;慕斯主页&#xff1a;修仙—别有洞天 ♈️今日夜电波&#xff1a;Nonsense—Sabrina Carpenter 0:50━━━━━━️&#x1f49f;──────── 2:43 &#x1f504; ◀️ ⏸ ▶️ …

Redis 持久化详解:RDB 与 AOF 的配置、触发机制和实际测试

什么是持久化&#xff1f; 就是 Redis 将内存数据持久化到硬盘&#xff0c;避免从数据库恢复数据。之所以避免从数据库恢复数据是因为后端数据通常有性能瓶颈&#xff0c;大量数据从数据库恢复可能会给数据库造成巨大压力。 Redis 持久化通常有 RDB 和 AOF 两种方式&#xff…

​ PaddleHub 首页图像 - 文字识别chinese_ocr_db_crnn_server​

PaddleHub 便捷地获取PaddlePaddle生态下的预训练模型&#xff0c;完成模型的管理和一键预测。配合使用Fine-tune API&#xff0c;可以基于大规模预训练模型快速完成迁移学习&#xff0c;让预训练模型能更好地服务于用户特定场景的应用 零基础快速开始WindowsLinuxMac Paddle…