[SCOI2009]生日礼物 双指针

题意:
小西有一条很长的彩带,彩带上挂着各式各样的彩珠。已知彩珠有N个,分为K种。简单的说,可以将彩带考虑为x轴,每一个彩珠有一个对应的坐标(即位置)。某些坐标上可以没有彩珠,但多个彩珠也可以出现在同一个位置上。 小布生日快到了,于是小西打算剪一段彩带送给小布。为了让礼物彩带足够漂亮,小西希望这一段彩带中能包含所有种类的彩珠。同时,为了方便,小西希望这段彩带尽可能短,你能帮助小西计算这个最短的长度么?彩带的长度即为彩带开始位置到结束位置的位置差。
题解:
双指针。

利用STL里面的pair容器,默认排序的性质,就可以不需要自己写一个自定义函数了。
因为宝石的种类很少,当我们每取一段范围时,我们暴力判断以下这个区间是否满足。
时间复杂度 O(nlogn+n*60) ?
双指针判断条件
如果这段区间的宝石满足 题目要求
那么我们r++,同时加上此位置的宝石数目。
如果满足了我们就l++,同时减去那个位置的宝石数目。

这里的区间5-8的区间长度是3,不是4 ,注意一下即可.
代码:

/*Keep on going Never give up*/
#pragma GCC optimize(3,"Ofast","inline")
#include
#include
#include 
#include
#include
#include
#include
#include
#include
#include 
#include

//#define int long long
using namespace std;
const int maxn = 100;
const int MaxN = 0x3f3f3f3f;
const int MinN = 0xc0c0c00c;
typedef long long ll;
const int inf=0x3f3f3f3f;
const ll mod=1e9+7;
vector <pair<int,int> > v;
int visited[100];
int n,k;
bool check(){
    for(int i=1;i<=k;i++){
        if(visited[i]==0) return false;
    }
    return true;
}

signed main(){
    cin>>n>>k;
    for(int i=1;i<=k;i++){
        int m;
        cin>>m;
        for(int j=0;j<m;j++){
            int x;
            scanf("%d",&x);
            v.push_back(make_pair(x,i));
        }
    }
    sort(v.begin(),v.end());
    int l=0,r=0;
    int ans=MaxN;
    //for(auto it:v) cout<
    visited[v[0].second]++;
    while(r<n){
        //cout<
        if(check()){
            ans=min(ans,v[r].first-v[l].first);
            visited[v[l].second]--;
            l++;
        }
        else{
            r++;
            visited[v[r].second]++;

        }
    }
    cout<<ans<<endl;
    return 0;
}

你可能感兴趣的:(牛客每日一题,牛客)