作业-[luogu4396][AHOI2013]-莫队
<题面>
卡常终究比不上算法的优化……
这是莫队的有点小坑的题,
首先不一定能想到,想到不一定打对,打对不一定打好。
首先你会发现,这个题的时限是很长的~
$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]-莫队的更多相关文章
- 洛谷P4396 作业 [AHOI2013] 莫队
正解:莫队 解题报告: 传送门! 天呐太久没做莫队了连板子都认不出来了,,,所以复健下做下莫队的题目QAQ 就很板子鸭,和莫队板子比好像只有一个离散化,,,?就不讲了QAQ 等下直接放代码QAQ ov ...
- 【Luogu4396】[AHOI2013]作业(莫队)
[Luogu4396][AHOI2013]作业(莫队) 题面 洛谷 题解 模板题 #include<iostream> #include<cstdio> #include< ...
- Bzoj 3236: [Ahoi2013]作业 莫队,分块
3236: [Ahoi2013]作业 Time Limit: 100 Sec Memory Limit: 512 MBSubmit: 1113 Solved: 428[Submit][Status ...
- BZOJ 3236: [Ahoi2013]作业( 莫队 + BIT )
莫队..用两个树状数组计算.时间复杂度应该是O(N1.5logN). 估计我是写残了...跑得很慢... ----------------------------------------------- ...
- BZOJ_3809_Gty的二逼妹子序列 && BZOJ_3236_[Ahoi2013]作业 _莫队+分块
BZOJ_3809_Gty的二逼妹子序列 && BZOJ_3236_[Ahoi2013]作业 _莫队+分块 Description Autumn和Bakser又在研究Gty的妹子序列了 ...
- 【BZOJ3809/3236】Gty的二逼妹子序列 [Ahoi2013]作业 莫队算法+分块
[BZOJ3809]Gty的二逼妹子序列 Description Autumn和Bakser又在研究Gty的妹子序列了!但他们遇到了一个难题. 对于一段妹子们,他们想让你帮忙求出这之内美丽度∈[a,b ...
- [AHOI2013]作业 (莫队+分块)
[AHOI2013]作业 (莫队+分块) 题面 给定了一个长度为n的数列和若干个询问,每个询问是关于数列的区间[l,r],首先你要统计该区间内大于等于a,小于等于b的数的个数,其次是所有大于等于a,小 ...
- 【分块,莫队】【P4396】【AHOI2013】作业
传送门 Description 此时己是凌晨两点,刚刚做了Codeforces的小A掏出了英语试卷.英语作业其实不算多,一个小时刚好可以做完.然后是一个小时可以做完的数学作业,接下来是分别都是一个小时 ...
- [AHOI2013]作业 & Gty的二逼妹子序列 莫队
---题面--- 题解: 题目要求统计一个区间内数值在[a, b]内的数的个数和种数,而这个是可以用树状数组统计出来的,所以可以考虑莫队. 考虑区间[l, r]转移到[l, r + 1],那么对于维护 ...
随机推荐
- [USACO2005 nov] Grazing on the Run【区间Dp】
Online Judge:bzoj1742,bzoj1694 Label:区间Dp 题目描述 John养了一只叫Joseph的奶牛.一次她去放牛,来到一个非常长的一片地,上面有N块地方长了茂盛的草.我 ...
- js正则笔记
//内容 var innerhtml = $('.reading_box_m').html().replace(/ tag="?[一二三四五六七八九十]+"?/ig, " ...
- 关于安装了sqlite对于vs的组件,重启vs后,在外面可以连接sqlite数据库,但是在建立实体模型时没有sqlite数据源的问题
出自:http://bbs.csdn.net/topics/390917337 兄弟,刚刚在stackoverflow上找到了解决方法了http://stackoverflow.com/questio ...
- BZOJ2226:[SPOJ5971]LCMSum
Description Given n, calculate the sum LCM(1,n) + LCM(2,n) + .. + LCM(n,n), where LCM(i,n) denotes t ...
- sqlite3-入门日记4-实现C++类封装
一.前言: 今天试了下如何用C++类实现接口封装,感觉蛮好 .用于封装的类主要有两个,SQLiteStatement类和SQLiteWrapper类,是一个老外写的.我看了下源码,主要是对C接口进 ...
- Android之TableLayout表格布局
1.相关属性 1.1.常用属性 android:collapseColumns 设置需要被隐藏的列的序列号 android:shrinkColumns 设置允许被收缩的列的序列号 android:st ...
- 黑裙辉DS918+安装错误码21,安装教程 重装需要重新制作启动盘
不然报错误码21
- 795. Number of Subarrays with Bounded Maximum
数学的方式 是对于所有的字符分成简单的三类 0 小于 L 1 LR 之间 2 大于R 也就是再求 不包含 2 但是包含1 的子数组个数 不包含2的子数组个数好求 对于连续的相邻的n个 非2类数 就有 ...
- ListCtrl使用指南
http://blog.csdn.net/bqw2008/article/details/2047489 Windows ListCtrl使用技巧1. ListCtrl 风格 LVS_IC ...
- 廖雪峰Java11多线程编程-1线程的概念-3线程的状态
1线程的状态 线程终止的的原因: run()或call()方法执行完成,线程正常结束 线程抛出一个未捕获的Exception或Error 直接调用该线程的stop()方法来结束该线程--该方法容易导致 ...