【精品】【算法实战】每日一题:如何用Python实现给定整数序列中寻找最小长度窗口以包含所有不同元素的算法?

问题:

如何用Python实现给定整数序列中寻找最小长度窗口以包含所有不同元素的算法?

核心思路

核心思路是利用双端队列(作为滑动窗口)来找到一个满足特定条件的最小长度子序列。算法遍历给定的序列,对于每个新数据点,执行以下三种操作之一:

  1. 如果数据点不在当前窗口中,则将其添加到队列中(表示窗口扩展)。
  2. 如果数据点已存在于窗口中,但队首和队尾的元素不同(意味着窗口中尚需包含此元素),则仍然将其加入队列。(损耗元素,知道有但是还是得加入)
  3. 如果数据点已存在,且队首和队尾的元素相同(表示该元素类型已满足),则从队列中移除队首元素(缩短窗口)。

在遍历过程中,维护一个记录最小长度窗口的变量。如果当前窗口包含所有必需的不同元素类型,并且比之前记录的任何窗口都小,则更新最小长度窗口的记录。最终,得到一个满足条件的最小长度窗口。

进一步给出伪码思路

复制代码
初始化一个双端队列 q 和一些计数器以及变量。

定义主函数 main:
    读取输入的 n 和 m,分别代表数组长度和目标不同元素的数量。
    读取数组 a 并将其转换为整数列表。
    对数组 a 中的每个元素执行以下操作:
        如果当前元素的计数器 cnt 为 0,增加类型数量 type。
        增加当前元素的计数器。
        将当前索引添加到双端队列 q 的末尾。
        如果双端队列 q 不为空且队首元素的计数器大于 1,移除队首元素,直到计数器为 1。
        如果当前类型数量等于目标 m 且双端队列的长度小于当前记录的最小长度 ans,则更新 ans 以及起始索引 l 和结束索引 r。
    打印出最小长度窗口的起始索引和结束索引,索引加 1(因为 Python 索引从 0 开始)。

定义函数 qsize 来返回双端队列 q 的大小。

如果这是主程序,调用 main 函数。

CODE

python 复制代码
from collections import deque

# 初始化变量
ans = float('inf')
n, m = 0, 0
a = []
cnt = [0] * 10000
l, r, type = 0, 0, 0
q = deque()

def main():
    global ans, l, r, type
    n, m = map(int, input().split())
    a = list(map(int, input().split()))
    
    # 统计每个数出现的次数,并判断是否需要type++
    for i in range(n):
        if cnt[a[i]] == 0:
            type += 1
        cnt[a[i]] += 1
        q.append(i)
        
        # 移除队首元素,直到cnt[a[q[0]]]为0
        while q and cnt[a[q[0]]] > 1:
            cnt[a[q.popleft()]] -= 1
            
        # 如果type=m,且队列长度<ans,则更新ans和l,r
        if type == m and qsize() < ans:
            ans = qsize()
            l = q[0]
            r = q[-1]
    
    print(l + 1, r + 1)  # 输出时加1,因为Python索引从0开始

def qsize():
    return len(q)

if __name__ == '__main__':
    main()

注:

  1. ans = float('inf'):
    ans 变量用于存储找到的满足条件的最小长度子数组的长度。初始化为正无穷大(float('inf')),意味着我们从找一个非常大的长度开始,随着算法的进行,如果找到更小的满足条件的子数组,就会更新这个值。

  2. n, m = 0, 0:
    n 是数组 a 的长度,即数组中元素的总数。
    m 是我们需要在子数组中找到的不同元素的类型数。

  3. a = []:
    a 是一个空列表,稍后将用来存储输入的整数数组。

  4. cnt = [0] * 10000:
    cnt 是一个长度为 3000 的列表,用于计数。它将用于跟踪数组 a 中每个元素的出现次数,10000因为测试用例可能比较大

  5. l, r, type = 0, 0, 0:
    lr 是子数组的起始和结束索引,初始化为 0。随着算法的进行,它们将被更新为满足条件的最小长度子数组的起始和结束索引。
    type 是一个计数器,用于跟踪当前窗口中不同元素的类型数。

  6. q = deque():
    q 是一个 deque(双端队列)实例,它将被用作滑动窗口的数据结构。双端队列允许我们从两端快速添加和删除元素,非常适合实现滑动窗口。

END

相关推荐
进来有惊喜3 分钟前
深度学习:迁移学习
python·深度学习
mit6.82414 分钟前
[贪心_7] 最优除法 | 跳跃游戏 II | 加油站
数据结构·算法·leetcode
keep intensify16 分钟前
通讯录完善版本(详细讲解+源码)
c语言·开发语言·数据结构·算法
shix .23 分钟前
2025年PTA天梯赛正式赛 | 算法竞赛,题目详解
数据结构·算法
风铃儿~32 分钟前
Java面试高频问题(26-28)
java·算法·面试
wuqingshun31415934 分钟前
蓝桥杯 4. 卡片换位
算法·职场和发展·蓝桥杯
江沉晚呤时36 分钟前
深入了解C# List集合及两种常见排序算法:插入排序与堆排序
windows·sql·算法·oracle·c#·排序算法·mybatis
@正在学习驰骋的小马40 分钟前
九、小白如何用Pygame制作一款跑酷类游戏(添加前进小动物作为动态障碍物)
python·游戏·pygame
BXCQ_xuan1 小时前
Django API 响应格式:一个新手踩坑记
python·django·状态模式
Python×CATIA工业智造1 小时前
基于CATIA参数化管道建模的自动化插件开发实践——NX建模之管道命令的参考与移植
python·pycharm·catia二次开发