题意:
Jasio有n个不同的玩具,它们都被放在了很高的架子上,地板上不会有超过k个玩具。当Jasio想玩地板上的其他玩具时,他会自己去拿,如果他想玩的玩具在架子上,他的妈妈则会帮他去拿,当她拿玩具的时候,顺便也会将一个地板上的玩具放上架子使得地板上有足够的空间。他的妈妈知道孩子的p个请求,求通过决定每次换玩具时换下哪个所能使他妈妈架子上拿玩具的次数的最小值。k≤100000,p≤500000。bzoj1826中的玩具种类需离散化。
题解:
首先求出每个请求的下一次对同种类玩具的请求的位置。之后维护一个优先队列求当前地板上玩具的下次请求位置,每次换玩具时把下次请求位置最大的弹出。
代码(1528):
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <queue>
#include <set>
#define inc(i,j,k) for(int i=j;i<=k;i++)
#define maxn 500010
#define INF 0x3fffffff
using namespace std; inline int read(){
char ch=getchar(); int f=,x=;
while(ch<''||ch>''){if(ch=='-')f=-; ch=getchar();}
while(ch>=''&&ch<='')x=x*+ch-'',ch=getchar();
return f*x;
}
struct nd{
int v,id; bool operator < (const nd&a)const{return v<a.v;}
};
priority_queue<nd>q;
int n,k,p,a[maxn],next[maxn],tot,ans,last[maxn]; bool zsye[maxn];
int main(){
n=read(); k=read(); p=read(); inc(i,,p)a[i]=read();
inc(i,,n)last[i]=p+; for(int i=p;i>=;i--)next[i]=last[a[i]],last[a[i]]=i;
inc(i,,p){
if(!zsye[a[i]]){
if(tot<k){
zsye[a[i]]=; ans++; tot++;
if(next[i])q.push((nd){next[i],a[i]});else q.push((nd){p+,a[i]});
}else{
zsye[a[i]]=; ans++; nd x=q.top(); q.pop(); zsye[x.id]=;
if(next[i])q.push((nd){next[i],a[i]});else q.push((nd){p+,a[i]});
}
}else{
q.push((nd){next[i],a[i]});
}
}
printf("%d",ans); return ;
}
20161109