<题面>

卡常终究比不上算法的优化……

这是莫队的有点小坑的题,

首先不一定能想到,想到不一定打对,打对不一定打好。


首先你会发现,这个题的时限是很长的~

$n$和$m$也是很大的。

于是我们可以计算一个时间复杂度,顺便搞一个块长。

首先有$\Theta (M)$的询问,设块长为$L$,

于是左端点移动,不跨块,移动不超过$L$

右端点移动$N$

然后就有左面的总共移动$M \times L $

右面的移动$\frac{N}{L}\times N$这里应该就是说,每次要挪一个 $N$,而左端点只挪$\frac{N}{L}$次。

(转移一会再算)

这样莫队总复杂度:$\Theta(M \times L + \frac{N^2}{L})$

所以当块长为$\sqrt{\frac{N^2}{M}}=\frac{N}{\sqrt{M}}$时复杂度是最优的(基本不等式)

然后我们想转移,他让我们求一个区间数的个数。

那么,我们就可以用个数据结构,来维护$[a,b]$中数的个数和去重后的数个数

仔细算下,$\log N$的时间复杂度还能承受。

所以选择树状数组或是线段树(常数有点大,用zkw好一些(但是用线段树的A的很困难~))

维护权值。

这样就可以通过用树状数组前缀和或是线段树区间查询解决转移问题。

总复杂度:$\Theta(N\sqrt{M}\log N)$

跑得还好说。

(有人说分块也行,大家加油咯!)

#include <algorithm>
#include <iostream>
#include <cstring>
#include <cstdio>
#include <cmath>
#define N 101101
#define LL long long //#include "debug.h" using namespace std;
struct QUERY{
int l,r,a,b,t;
LL ans1,ans2;
}query[N*10];
int num[N];
int pre[N],n,qn,sqn;
int inpart[N];
int tot[N],pretot[N];
inline int lowbit (int x){ return x&(-x); }
inline int getpart(int x){ return (x-1)/sqn+1; }
LL sum (int x){
LL s=0;
for(int i=x;i;i-=lowbit(i)) s+=pre[i];
return s;
}
LL gettot (int x){
LL s=0;
for(int i=x;i;i-=lowbit(i)) s+=pretot[i];
return s;
}
void chatot (int pos,int x){
while(pos<=n){
pretot[pos]+=x;
pos+=lowbit(pos);
}
}
void change (int pos,int x){
if(tot[pos]==1&&x==-1)
chatot(pos,-1);
else if(tot[pos]==0&&x== 1)
chatot(pos,1);
tot[pos]+=x;
while(pos<=n){
pre[pos]+=x;
pos+=lowbit(pos);
}
}
//bool CMP (const QUERY &a,const QUERY &b){
// if(inpart[a.l]==inpart[b.r])
// return a.r<b.r;
// return a.l<a.l;
//}
bool CMP(const QUERY &x,const QUERY &y){
return inpart[x.l]<inpart[y.l]||(inpart[x.l]==inpart[y.l]&&(inpart[x.l]&1?x.r<y.r:x.r>y.r));
}
bool Cmp (const QUERY &a,const QUERY &b){
return a.t<b.t;
}
int main (){
int a,b,c,d;
scanf("%d%d",&n,&qn);sqn=sqrt(1ll*n*n/qn)+1;
for(int i=1;i<=n;i++){
scanf("%d",num+i);
inpart[i]=getpart(i);
}
for(int i=1;i<=qn;i++){
scanf("%d%d%d%d",&a,&b,&c,&d);
query[i].l=a,query[i].r=b,query[i].t=i;
query[i].a=c,query[i].b=d;
}
sort(query+1,query+qn+1,CMP);
//for(int i=1;i<=qn;i++)cout<<query[i].l<<" "<<query[i].r<<endl;
int l=1,r=1,ql,qr;
change(num[1],1);
for(int i=1;i<=qn;i++){//pour(tot,1,n,3,"Tot");
ql=query[i].l,qr=query[i].r,
a =query[i].a,b =query[i].b;
while(l<ql)change(num[l] ,-1),l++;
while(l>ql)change(num[l-1], 1),l--;
while(r<qr)change(num[r+1], 1),r++;
while(r>qr)change(num[r] ,-1),r--;
query[i].ans1=sum(b)-sum(a-1);
query[i].ans2=gettot(b)-gettot(a-1);
}
sort(query+1,query+qn+1,Cmp);
for(int i=1;i<=qn;i++)
printf("%lld %lld\n",query[i].ans1,query[i].ans2);
return 0;
}

作业-[luogu4396][AHOI2013]-莫队的更多相关文章

  1. 洛谷P4396 作业 [AHOI2013] 莫队

    正解:莫队 解题报告: 传送门! 天呐太久没做莫队了连板子都认不出来了,,,所以复健下做下莫队的题目QAQ 就很板子鸭,和莫队板子比好像只有一个离散化,,,?就不讲了QAQ 等下直接放代码QAQ ov ...

  2. 【Luogu4396】[AHOI2013]作业(莫队)

    [Luogu4396][AHOI2013]作业(莫队) 题面 洛谷 题解 模板题 #include<iostream> #include<cstdio> #include< ...

  3. Bzoj 3236: [Ahoi2013]作业 莫队,分块

    3236: [Ahoi2013]作业 Time Limit: 100 Sec  Memory Limit: 512 MBSubmit: 1113  Solved: 428[Submit][Status ...

  4. BZOJ 3236: [Ahoi2013]作业( 莫队 + BIT )

    莫队..用两个树状数组计算.时间复杂度应该是O(N1.5logN). 估计我是写残了...跑得很慢... ----------------------------------------------- ...

  5. BZOJ_3809_Gty的二逼妹子序列 && BZOJ_3236_[Ahoi2013]作业 _莫队+分块

    BZOJ_3809_Gty的二逼妹子序列 && BZOJ_3236_[Ahoi2013]作业 _莫队+分块 Description Autumn和Bakser又在研究Gty的妹子序列了 ...

  6. 【BZOJ3809/3236】Gty的二逼妹子序列 [Ahoi2013]作业 莫队算法+分块

    [BZOJ3809]Gty的二逼妹子序列 Description Autumn和Bakser又在研究Gty的妹子序列了!但他们遇到了一个难题. 对于一段妹子们,他们想让你帮忙求出这之内美丽度∈[a,b ...

  7. [AHOI2013]作业 (莫队+分块)

    [AHOI2013]作业 (莫队+分块) 题面 给定了一个长度为n的数列和若干个询问,每个询问是关于数列的区间[l,r],首先你要统计该区间内大于等于a,小于等于b的数的个数,其次是所有大于等于a,小 ...

  8. 【分块,莫队】【P4396】【AHOI2013】作业

    传送门 Description 此时己是凌晨两点,刚刚做了Codeforces的小A掏出了英语试卷.英语作业其实不算多,一个小时刚好可以做完.然后是一个小时可以做完的数学作业,接下来是分别都是一个小时 ...

  9. [AHOI2013]作业 & Gty的二逼妹子序列 莫队

    ---题面--- 题解: 题目要求统计一个区间内数值在[a, b]内的数的个数和种数,而这个是可以用树状数组统计出来的,所以可以考虑莫队. 考虑区间[l, r]转移到[l, r + 1],那么对于维护 ...

随机推荐

  1. NuGet 命令行使用EntityFrameWork

    初始化 Enable-migrations 迁移 Add-Migration Donator_Add_CreationTime 执行操作 UpDate-database 撤销更改 Update-Dat ...

  2. 用在 AMD64 上 aria2_1.33.1-1_amd64.deb 的下载页面

    用在 AMD64 上 aria2_1.33.1-1_amd64.deb 的下载页面 如果您正在运行 Ubuntu,请尽量使用像 aptitude 或者 synaptic 一样的软件包管理器,代替人工手 ...

  3. 廖雪峰Java12maven基础-1maven入门-3构建流程

    maven是一个Java项目管理和构建工具: 标准化项目结构 标准化构建流程(编译.打包.发布) 依赖管理 Maven的构建流程 clean 删除所有编译生成的文件 compile 编译源码.编译测试 ...

  4. 网络编程(client发信息给server)

    client发信息给server

  5. day08 网络设置、软件包管理

    网络设置 ifconfig //最小化安装时不可用,需要安装安装包,命令为查看网卡信息 yum install net-tools mtu 网卡的最大发送字节 iptables -F 清掉防火墙配置 ...

  6. 2-sat——暴力染色输出方案hdu1814

    因为要求输出字典序最小的解,所以用暴力染色 具体有点像二分图染色 遍历0-2*n-1个点,尝试将每个点染成1,该点所能到达的所有点都要染成1 如果不行,则把上该点的影响消除,再把对立点染成1,如果还不 ...

  7. 集合划分——cf1028D思维题

    非常思维的一道题目,题意很长 给定s1,s2两个集合,s1维护最大值,s2维护最小值,s1的所有元素要比s2小 操作1:往两个集合里的任意一个添加x 操作2:把x从所在的集合里删掉:要求被删的x必须是 ...

  8. BCB如何编写,调用动态链接库DLL

    一 编写动态链接库DLL DLL简称动态链接库,是Windows中程序的重要组成部分.想象一下,一个程序需要多人共同完成开发,怎么个共同法?这时我们就要考虑把程序分为好几个模块,团队每一个成员开发一个 ...

  9. javaweb的几种开发模式

    1.MVC模式基础 1.1.MVC模式简介 MVC是一种架构型模式,它本身并不引入新的功能,只是用来指导我们改善应用程序的架构,使得应用的模型和视图相分离,从而达到更好的开发和维护效率.在MVC模式中 ...

  10. 转:链表相交有环 经典面试题(三)附答案 算法+数据结构+代码 微软Microsoft、谷歌Google、百度、腾讯

    源地址:http://blog.csdn.net/sj13051180/article/details/6754228 1.判断单链表是否有环,要求空间尽量少(2011年MTK) 如何找出环的连接点在 ...