塔子哥的最短区间-小米2023笔试(codefun2000)

题目链接
塔子哥的最短区间-小米2023笔试(codefun2000)

题目内容

塔子哥有一个长度为 n 的数组 a 和 长度为 m 的数组 b ,下标均从 1 开始。
现在,塔子哥想让你找出一个最短的区间 l,r , 这个区间中数 x 的数量至少出现了 b[x] 次。

输入描述

第一行,两个整数 n,m( 1 ≤ n , m ≤ 1 0 5 1≤n,m≤10^5 1n,m105 ) 分别表示数组 a 和数组 b 的长度。
第二行,n 个整数表示数组 a 。
第三行,m 个整数表示数组 b 。

输出描述

一个整数,表示最短区间的长度,如果不存在,则输出 -1 。

样例1

输入

6 4
1 1 4 5 1 4
2 0 0 2

输出

5

提示

区间 [2,6] 满足 1 和 4 出现了至少两次,2 和 3 出现了至少 0 次。 可以证明没有更短的区间满足了。

题解1

#include<bits/stdc++.h>
using namespace std;

const int N = 1e5 + 10;

int n, m, a[N], b[N], cnt[N]; // cnt[i]表示i出现的次数 

bool check(int x){ // 当前枚举的区间长度为x 
	memset(cnt, 0, sizeof cnt);
	for(int i = 1; i <= n; i++){ 
		cnt[a[i]]++;// 双指针
		if(i >= x){  
			int j = 1;
			while(j <= m && cnt[j] >= b[j]) j++;
			if(j > m) return true;
			cnt[a[i - x + 1]]--;
		}
	}
	return false;
}
int main(){
	scanf("%d%d", &n, &m);
	for(int i = 1; i <= n; i++) scanf("%d", &a[i]);
	for(int i = 1; i <= m; i++) scanf("%d", &b[i]);
	
	int left = - 1, right = n + 1, mid;
	while(left + 1 < right){ // 二分枚举满足条件的最小区间的长度 
		mid = (left + right)/2;
		if(check(mid)) right = mid;
		else left = mid;
	}
	printf("%d\n", check(right)?right:-1);
	return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值