【BZOJ2527】MET-Meteors(整体二分)

17 篇文章 0 订阅

题面

BZOJ权限题,良心洛谷链接

题解

其实我也不会做
看了zsy博客才会做。。。

这题如果直接爆算做显然行不通
如果只有单次询问,我们就可以二分答案
但是询问太多。。
不会二分。。

怎么办?
我们来想想瓶颈在哪里
如果每次都进行一次单次二分
我们就需要不停的计算前缀和
但是其实再进行别的二分的时候我们已经算过了
这里就算重了

怎么解决?
那我们就不让他算重:
我们把所有询问一起二分
这样每次计算完之后
所有的询问就被分成了两部分:
一部分是当前二分值有解
另一部分是无解
这样就可以解决重复计算的问题了

因为要反复计算前缀和
所以使用树状数组来计算就行了

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<set>
#include<map>
#include<vector>
#include<queue>
using namespace std;
#define ll long long
#define RG register
#define MAX 320000
inline int read()
{
    RG int x=0,t=1;RG char ch=getchar();
    while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
    if(ch=='-')t=-1,ch=getchar();
    while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
    return x*t;
}
struct Rain{int l,r;ll w;}r[MAX];
ll c[MAX];
int n,m,ans[MAX],K;
int lowbit(int x){return x&(-x);}
void Add(int x,int w){while(x<=m)c[x]+=w,x+=lowbit(x);}
ll getsum(int x){ll ret=0;while(x)ret+=c[x],x-=lowbit(x);return ret;}
vector<int> a[MAX];
struct Query{int id;ll p;}p[MAX],p1[MAX],p2[MAX];
void Fall(int t,int opt)
{
    if(r[t].l>r[t].r)Add(1,opt*r[t].w);
    Add(r[t].l,opt*r[t].w),Add(r[t].r+1,-opt*r[t].w);
}
void Work(int l,int r,int L,int R)
{
    if(L>R)return;
    if(l==r)
    {
        for(int i=L;i<=R;++i)ans[p[i].id]=l;
        return;
    }
    int mid=(l+r)>>1;
    ll tot;
    int t1=0,t2=0;
    for(int i=l;i<=mid;++i)Fall(i,1);
    for(int i=L;i<=R;++i)
    {
        tot=0;
        for(int j=0,l=a[p[i].id].size();j<l;++j)tot+=getsum(a[p[i].id][j]);
        if(tot>=p[i].p)p1[++t1]=p[i];
        else p[i].p-=tot,p2[++t2]=p[i];
    }
    for(int i=l;i<=mid;++i)Fall(i,-1);
    for(int i=1;i<=t1;++i)p[i+L-1]=p1[i];
    for(int i=1;i<=t2;++i)p[i+L+t1-1]=p2[i];
    Work(l,mid,L,L+t1-1);
    Work(mid+1,r,L+t1,R);
}
int main()
{
    n=read();m=read();
    for(int i=1;i<=m;++i)a[read()].push_back(i);
    for(int i=1;i<=n;++i)p[i].p=read(),p[i].id=i;
    K=read();
    for(int i=1;i<=K;++i)r[i].l=read(),r[i].r=read(),r[i].w=read();
    ++K;r[K]=(Rain){1,m,1e9};
    Work(1,K,1,n);
    for(int i=1;i<=n;++i)ans[i]==K?puts("NIE"):printf("%d\n",ans[i]);
    return 0;
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
BZOJ 2908 题目是一个数据下载任务。这个任务要求下载指定的数据文件,并统计文件中小于等于给定整数的数字个数。 为了完成这个任务,首先需要选择一个合适的网址来下载文件。我们可以使用一个网络爬虫库,如Python中的Requests库,来帮助我们完成文件下载的操作。 首先,我们需要使用Requests库中的get()方法来访问目标网址,并将目标文件下载到我们的本地计算机中。可以使用以下代码实现文件下载: ```python import requests url = '目标文件的网址' response = requests.get(url) with open('本地保存文件的路径', 'wb') as file: file.write(response.content) ``` 下载完成后,我们可以使用Python内置的open()函数打开已下载的文件,并按行读取文件内容。可以使用以下代码实现文件内容读取: ```python count = 0 with open('本地保存文件的路径', 'r') as file: for line in file: # 在这里实现对每一行数据的判断 # 如果小于等于给定整数,count 加 1 # 否则,不进行任何操作 ``` 在每一行的处理过程中,我们可以使用split()方法将一行数据分割成多个字符串,并使用int()函数将其转换为整数。然后,我们可以将该整数与给定整数进行比较,以判断是否小于等于给定整数。 最后,我们可以将统计结果打印出来,以满足题目的要求。 综上所述,以上是关于解决 BZOJ 2908 数据下载任务的简要步骤和代码实现。 希望对您有所帮助。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值