BZOJ 2527 [Poi2011]Meteors(整体二分)

【题目链接】 http://www.lydsy.com/JudgeOnline/problem.php?id=2527

【题目大意】

  有N个成员国。现在它发现了一颗新的星球,
  这颗星球的轨道被分为M份(第M份和第1份相邻),第i份上有第Ai个国家的太空站。
  这个星球经常会下陨石雨。BIU已经预测了接下来K场陨石雨的情况。
  BIU的第i个成员国希望能够收集Pi单位的陨石样本。
  你的任务是判断对于每个国家,它需要在第几次陨石雨之后,才能收集足够的陨石。

【题解】

  如果枚举每场陨石雨,树状数组查询每个国家是否完成收集,复杂度O(Kmlogm),难以接受
  因此我们对答案整体二分,不同位置的答案不断划分到对应的陨石时段计算。
  将完成任务的国家放到左边区间,未达成的放到右边区间。复杂度O(mlogmlogK).

【代码】

#include <cstdio>
#include <cstring>
#include <vector>
using namespace std;
const int N=300010;
typedef long long LL;
int x,n,m,T,K,l[N],r[N],val[N],id[N],p[N],tmp[N],ans[N],mark[N];
LL c[N];
vector<int> v[N];
LL query(int x){LL s=0;while(x)s+=c[x],x-=x&-x;return s;}
void add(int x,int val){while(x<=m)c[x]+=val,x+=x&-x;}
void update(int x,int k){
    if(l[x]<=r[x])add(l[x],k*val[x]),add(r[x]+1,-k*val[x]);
    else{
        add(1,k*val[x]),add(r[x]+1,-k*val[x]);
        add(l[x],k*val[x]);
    }
}
void solve(int l,int r,int L,int R){
    if(l>r)return;
    if(L==R){for(int i=l;i<=r;i++)ans[id[i]]=L;return;}
    int mid=(L+R)>>1;
    while(T<=mid)update(++T,1);
    while(T>mid)update(T--,-1);
    int cnt=0,x; LL tot;
    for(int i=l;i<=r;i++){
        tot=0; x=id[i];
        for(int j=0;j<v[x].size();j++){
            tot+=query(v[x][j]);
            if(tot>=p[x])break;
        }if(tot>=p[x])mark[x]=1,cnt++;
        else mark[x]=0;
    }int l1=l,l2=l+cnt;
    for(int i=l;i<=r;i++){
        if(mark[id[i]])tmp[l1++]=id[i];
        else tmp[l2++]=id[i];
    }for(int i=l;i<=r;i++)id[i]=tmp[i];
    solve(l,l1-1,L,mid);
    solve(l1,l2-1,mid+1,R);
}
int main(){
    while(~scanf("%d%d",&n,&m)){
        for(int i=1;i<=m;i++)scanf("%d",&x),v[x].push_back(i);
        for(int i=1;i<=n;i++)scanf("%d",&p[i]);
        scanf("%d",&K);
        for(int i=1;i<=K;i++)scanf("%d%d%d",&l[i],&r[i],&val[i]);
        l[++K]=1;r[K]=m;val[K]=0x3f3f3f3f;
        for(int i=1;i<=n;i++)id[i]=i;
        solve(1,n,1,K);
        for(int i=1;i<=n;i++){
            if(ans[i]==K)puts("NIE");
            else printf("%d\n",ans[i]);
        }
    }return 0;
}
时间: 2024-10-13 22:28:32

BZOJ 2527 [Poi2011]Meteors(整体二分)的相关文章

BZOJ 2527 Poi2011 Meteors 整体二分+线段树 / 可持久化线段树(MLE)

题目大意:给定一个环,每个节点有一个所属国家,k次事件,每次对[l,r]区间上的每个点点权加上一个值,求每个国家最早多少次操作之后所有点的点权和能达到一个值 首先我们考虑暴力想法 对于每个国家分开讨论 二分操作次数 但是这样每次Judge的时候我们要模拟1~mid所有的操作 浪费在这里的复杂度实在太大 这样做每个国家需要模拟O(klogk)次操作 时间复杂度O(nklogk) TLE 我们需要对浪费在这里的复杂度做一些改进 1.可持久化线段树(MLE) 每次二分一个mid之后 我们要找到mid次

【bzoj2527】[Poi2011]Meteors 整体二分+树状数组

题目描述 有N个成员国.现在它发现了一颗新的星球,这颗星球的轨道被分为M份(第M份和第1份相邻),第i份上有第Ai个国家的太空站. 这个星球经常会下陨石雨.BIU已经预测了接下来K场陨石雨的情况.BIU的第i个成员国希望能够收集Pi单位的陨石样本.你的任务是判断对于每个国家,它需要在第几次陨石雨之后,才能收集足够的陨石. 输入 第一行是两个数N,M. 第二行有M个数,第i个数Oi表示第i段轨道上有第Oi个国家的太空站. 第三行有N个数,第i个数Pi表示第i个国家希望收集的陨石数量. 第四行有一个

Luogu3527 POI2011 Meteors 整体二分、树状数组、差分

传送门 比较板子的整体二分题目,时限有点紧注意常数 整体二分的过程中将时间在\([l,mid]\)之间的流星使用树状数组+差分进行维护,然后对所有国家查看一遍并分好类,递归下去,记得消除答案在\([mid+1,r]\)的询问中时间在\([l,mid]\)的流星操作的贡献 注意:可能存在某一段时间某一个国家的流星数量超过long long范围,应该当某个时候国家流星量和大于等于国家需求值时直接退出,这样可以避免这个问题. #include<bits/stdc++.h> #define INF 0

【BZOJ-2527】Meteors 整体二分 + 树状数组

2527: [Poi2011]Meteors Time Limit: 60 Sec  Memory Limit: 128 MBSubmit: 831  Solved: 306[Submit][Status][Discuss] Description Byteotian Interstellar Union (BIU) has recently discovered a new planet in a nearby galaxy. The planet is unsuitable for colo

2527: [Poi2011]Meteors

2527: [Poi2011]Meteors Time Limit: 60 Sec  Memory Limit: 128 MBSubmit: 1528  Solved: 556[Submit][Status][Discuss] Description Byteotian Interstellar Union (BIU) has recently discovered a new planet in a nearby galaxy. The planet is unsuitable for col

bzoj 2527 Meteors - 整体二分 - 树状数组

Description Byteotian Interstellar Union (BIU) has recently discovered a new planet in a nearby galaxy. The planet is unsuitable for colonisation due to strange meteor showers, which on the other hand make it an exceptionally interesting object of st

BZOJ 2738 矩阵乘法 整体二分+二维树状数组

题目大意:给定一个矩阵,多次求某个子矩阵中的第k小 分块解法见 http://blog.csdn.net/popoqqq/article/details/41356899 <论除最小割外题目解法从来与题目名称无关系列> 整体二分 Solve(x,y,S)表示处理答案在[x,y]区间内的询问集合S 预先将所有数按照大小排序 每次将[1,mid]之间的数插入树状数组 然后对于分治内部的每一个询问 去树状数组中查询相应子矩阵的数值 如果小于等于k就划分到左集合S1 否则划分到右集合S2 然后Solv

Luogu P3527 [POI2011]MET-Meteors 整体二分

思路:整体二分 提交:4次 错因:树状数组开的$int$ 题解: 二分操作序列,将仅用$[l,md]$即可满足要求的国家递归到左半边,将仅用$[l,md]$不能满足要求的国家,把他们的要求去掉左半边的贡献,递归到右半边. 具体来说,开一个以空间站为下标的树状数组(把环展成链),区间加单点求和转化为差分和前缀和,依次加入$[l,md]$中的所有操作区间: 然后每个国家枚举自己的所有空间站,计算贡献,判断前$[l,md]$是否满足,来决定向左右递归的方向. #include<iostream> #

bzoj 2525: [Poi2011]Dynamite【二分+树上贪心】

一眼二分.然后重点是树上贪心部分 长得像dp一样,设mn为子树内已炸点的最浅点,mx为子树内没有炸并且需要炸的最深点,然后转移直接从子树继承即可 然后是判断当前u点是否需要炸,当mx[u]+mn[u]<=mid,当前子树可以自己消化,所以mx[u]=-inf:否则,就需要在u炸一下 #include<iostream> #include<cstdio> using namespace std; const int N=300005; int n,m,h[N],cnt,d[N]