Description
小西有一条很长的彩带,彩带上挂着各式各样的彩珠。已知彩珠有N个,分为K种。简单的说,可以将彩带考虑为x轴,每一个彩珠有一个对应的坐标(即位置)。某些坐标上可以没有彩珠,但多个彩珠也可以出现在同一个位置上。 小布生日快到了,于是小西打算剪一段彩带送给小布。为了让礼物彩带足够漂亮,小西希望这一段彩带中能包含所有种类的彩珠。同时,为了方便,小西希望这段彩带尽可能短,你能帮助小西计算这个最短的长度么?彩带的长度即为彩带开始位置到结束位置的位置差。
题解:
十分简单,独自想出来的.
题目中要求最短的一段区间,使得该区间有所有的色块种类.
考虑枚举右端点,设该区间左端点的色块为 $i$,那么 $i$ 一定是距离右端点最近的一个 $i$ ,即 $i$ 的位置是从右端点开始数起 $i$ 首次出现的位置.
证明:
如果不是最近的,那么可以将该区间缩短,因为 $i$ 在距离右端点更近的位置有替代者.
色块只有 60 个,考虑将坐标离散化暴力转移即可,途中顺便维护距离每一个坐标距离最远且第一次出现的距离,所有点中该值的最小值就是答案.
直接开数组会炸掉,滚动优化一下即可.
Code:
#include<bits/stdc++.h>
#define setIO(s) freopen(s".in","r",stdin)
#define maxn 1010000
#define inf 0x7fffffff
using namespace std;
void Min(int &a,int b)
{
if(b<a) a=b;
}
void Max(int &a,int b)
{
if(b>a) a=b;
}
struct Node
{
int id,d;
Node(int a=0,int b=0):id(id),d(d){}
}nodes[maxn];
bool cmp(Node a,Node b)
{
return a.d<b.d;
}
int arr[maxn],lst[2][61];
int main()
{
// setIO("input");
int n,k,m,cnt=0;
scanf("%d%d",&n,&k);
for(int i=1;i<=k;++i)
{
int t,x;
scanf("%d",&t);
for(int j=1;j<=t;++j)
{
scanf("%d",&x);
nodes[++cnt].id=i, nodes[cnt].d=x;
}
}
sort(nodes+1,nodes+1+n,cmp);
for(int i=1;i<=n;++i) arr[i]=nodes[i].d;
for(int j=0;j<=60;++j) lst[0][j]=lst[1][j]=inf;
for(int i=1;i<=n;++i)
{
nodes[i].d=lower_bound(arr+1,arr+1+n,nodes[i].d)-arr;
}
int ans=inf, cur=0;
for(int i=1,j;i<=n;i=j+1)
{
j=i;
int posl=nodes[i-1].d;
int posc=nodes[i].d;
int rec=0;
while(nodes[j+1].d==nodes[i].d) ++j;
for(int jj=i;jj<=j;++jj) lst[cur][nodes[jj].id]=0;
for(int jj=1;jj<=k;++jj)
{
if(lst[cur^1][jj]!=inf)
{
Min(lst[cur][jj],lst[cur^1][jj]+arr[posc]-arr[posl]);
}
rec=max(rec,lst[cur][jj]);
}
for(int j=0;j<=60;++j) lst[cur^1][j]=inf;
cur^=1;
Min(ans,rec);
}
printf("%d",ans);
return 0;
}