bzoj 1293: [SCOI2009]生日礼物 问题转化 + 性质分析 + 滚动数组优化

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; 
}

  

posted @ 2019-05-22 16:02  EM-LGH  阅读(154)  评论(0编辑  收藏  举报