COGS.1822.[AHOI2013]作业(莫队 树状数组/分块)
题目链接: COGS、BZOJ3236
Upd: 树状数组实现的是单点加 区间求和,采用值域分块可以\(O(1)\)修改\(O(sqrt(n))\)查询。同BZOJ3809.
莫队为\(O(n^{1.5})\)次修改和\(O(n)\)次查询。
注意这两个需求并不平衡,所以在搭配数据结构时常使用分块而不是线段树。
(转自莫队复杂度分析 by Meiku Kazami)
1.莫队+树状数组
/*
每个[l,r]的询问中又多了[a,b]值的限制。原先now是所有种类的个数,所以用 莫队+树状数组做
两问,维护两个树状数组
O(m*sqrt(n)*logn)
*/
#include<cmath>
#include<cstdio>
#include<cctype>
#include<algorithm>
using namespace std;
const int N=1e5+5,M=1e6+5;
int n,m,size,A[N],t1[N],t2[N],times[N],Ans1[M],Ans2[M];
struct Ques
{
int l,r,a,b,id;
bool operator <(const Ques &x)const
{
return l/size==x.l/size ? r<x.r : l/size<x.l/size;
}
}q[M];
inline int read()
{
int now=0,f=1;register char c=getchar();
for(;!isdigit(c);c=getchar())
if(c=='-') f=-1;
for(;isdigit(c);now=now*10+c-'0',c=getchar());
return now*f;
}
inline int lb(int x)
{
return x&-x;
}
void Update(int p,int v,int *t)
{
while(p<=n)
t[p]+=v, p+=lb(p);
}
int Query(int p,int *t)
{
int res=0;
while(p)
res+=t[p], p-=lb(p);
return res;
}
void Add(int p)
{
Update(A[p],1,t1);
if(!times[A[p]]) Update(A[p],1,t2);
++times[A[p]];
}
void Subd(int p)
{
Update(A[p],-1,t1);
--times[A[p]];
if(!times[A[p]]) Update(A[p],-1,t2);
}
int main()
{
freopen("ahoi2013_homework.in","r",stdin);
freopen("ahoi2013_homework.out","w",stdout);
n=read(),m=read();
size=sqrt(n);
for(int i=1;i<=n;++i)
A[i]=read();
for(int i=1;i<=m;++i)
q[i].l=read(), q[i].r=read(), q[i].a=read(), q[i].b=read(), q[i].id=i;
sort(q+1,q+1+m);
for(int l=1,r=0,i=1;i<=m;++i)
{
int ln=q[i].l,rn=q[i].r;
while(l<ln) Subd(l++);
while(l>ln) Add(--l);
while(r<rn) Add(++r);
while(r>rn) Subd(r--);
Ans1[q[i].id]=Query(q[i].b,t1)-Query(q[i].a-1,t1),//注意值的区间是[a,b]不是[l,r]
Ans2[q[i].id]=Query(q[i].b,t2)-Query(q[i].a-1,t2);
// printf("%d:%d %d ans1:%d ans2:%d\n",q[i].id,ln,rn,Ans1[q[i].id],Ans2[q[i].id]);
}
for(int i=1;i<=m;++i)
printf("%d %d\n",Ans1[i],Ans2[i]);
fclose(stdin);fclose(stdout);
return 0;
}
2.莫队+值域分块
#include <cmath>
#include <cstdio>
#include <cctype>
#include <algorithm>
#define gc() (SS==TT&&(TT=(SS=IN)+fread(IN,1,MAXIN,stdin),SS==TT)?EOF:*SS++)
const int N=1e5+5,MAXIN=2e6;
int n,m,size,Ans1[N*10],Ans2[N*10],A[N],tm[N],bel[N],sum1[500],sum2[500];
char IN[MAXIN],*SS=IN,*TT=IN;
struct Ask
{
int l,r,a,b,id;
bool operator <(const Ask &a)const
{
return l/size==a.l/size ? r<a.r : l/size<a.l/size;//更快 WTF
// return l/size==a.l/size?((l-1)/size&1 ? r>a.r : r<a.r):l/size<a.l/size;
}
}q[N*10];
inline int read()
{
int now=0;register char c=gc();
for(;!isdigit(c);c=gc());
for(;isdigit(c);now=now*10+c-'0',c=gc());
return now;
}
int Query(int *s,int l,int r,bool f)
{
int res=0,tmp=std::min(r,bel[l]*size);
for(int i=l; i<=tmp; ++i) res+= f?(tm[i]>0):tm[i];
if(bel[l]!=bel[r])
for(int i=(bel[r]-1)*size+1; i<=r; ++i)
res+= f?(tm[i]>0):tm[i];
for(int i=bel[l]+1; i<bel[r]; ++i) res+=s[i];
return res;
}
void Add(int p)
{
if(++tm[p]==1) ++sum2[bel[p]];
++sum1[bel[p]];
}
void Subd(int p)
{
if(!--tm[p]) --sum2[bel[p]];
--sum1[bel[p]];
}
int main()
{
n=read(), m=read(), size=sqrt(n);;//size=n/sqrt(m*2/3) //也没有更快 数组大小还要注意
for(int i=1; i<=n; ++i) bel[i]=(i-1)/size+1, A[i]=read();
for(int i=1; i<=m; ++i) q[i].l=read(),q[i].r=read(),q[i].a=read(),q[i].b=read(),q[i].id=i;
std::sort(q+1,q+1+m);
for(int l=1,r=0,ln,rn,i=1; i<=m; ++i)
{
int ln=q[i].l,rn=q[i].r;
while(l<ln) Subd(A[l++]);
while(l>ln) Add(A[--l]);
while(r<rn) Add(A[++r]);
while(r>rn) Subd(A[r--]);
Ans1[q[i].id]=Query(sum1,q[i].a,q[i].b,0), Ans2[q[i].id]=Query(sum2,q[i].a,q[i].b,1);
}
for(int i=1; i<=m; ++i) printf("%d %d\n",Ans1[i],Ans2[i]);
return 0;
}
COGS.1822.[AHOI2013]作业(莫队 树状数组/分块)的更多相关文章
- BZOJ3236[Ahoi2013]作业——莫队+树状数组/莫队+分块
题目描述 输入 输出 样例输入 3 4 1 2 2 1 2 1 3 1 2 1 1 1 3 1 3 2 3 2 3 样例输出 2 2 1 1 3 2 2 1 提示 N=100000,M=1000000 ...
- BZOJ 3236: [Ahoi2013]作业(莫队+树状数组)
传送门 解题思路 莫队+树状数组.把求\([a,b]\)搞成前缀和形式,剩下的比较裸吧,用\(cnt\)记一下数字出现次数.时间复杂度\(O(msqrt(n)log(n)\),莫名其妙过了. 代码 # ...
- [AHOI2013]作业 莫队 树状数组
#include<cmath> #include<cstdio> #include<algorithm> #include<string> #inclu ...
- bzoj3236 作业 莫队+树状数组
莫队+树状数组 #include<cstdio> #include<cstring> #include<iostream> #include<algorith ...
- BZOJ 3236 AHOI 2013 作业 莫队+树状数组
BZOJ 3236 AHOI 2013 作业 内存限制:512 MiB 时间限制:10000 ms 标准输入输出 题目类型:传统 评测方式:文本比较 题目大意: 此时己是凌晨两点,刚刚做了Co ...
- BZOJ_3289_Mato的文件管理_莫队+树状数组
BZOJ_3289_Mato的文件管理_莫队+树状数组 Description Mato同学从各路神犇以各种方式(你们懂的)收集了许多资料,这些资料一共有n份,每份有一个大小和一个编号 .为了防止他人 ...
- bzoj 3289: Mato的文件管理 莫队+树状数组
3289: Mato的文件管理 Time Limit: 40 Sec Memory Limit: 128 MB[Submit][Status][Discuss] Description Mato同学 ...
- 51nod 1290 Counting Diff Pairs | 莫队 树状数组
51nod 1290 Counting Diff Pairs | 莫队 树状数组 题面 一个长度为N的正整数数组A,给出一个数K以及Q个查询,每个查询包含2个数l和r,对于每个查询输出从A[i]到A[ ...
- 【BZOJ3460】Jc的宿舍(树上莫队+树状数组)
点此看题面 大致题意: 一棵树,每个节点有一个人,他打水需要\(T_i\)的时间,每次询问两点之间所有人去打水的最小等待时间. 伪·强制在线 这题看似强制在线,但实际上,\(pre\ mod\ 2\) ...
随机推荐
- ARMV8 datasheet学习笔记3:AArch64应用级体系结构之Synchronization and semapores
1.前言 本文主要介绍原子变量的实现原理,对原子变量的修改有一套特殊的机制 2. Local monitor和Global monitor UP时执行Load EX和Store EX时仅需关注Loca ...
- 配置samba文件服务器
1.打开"终端窗口",输入"sudo apt-get update"-->回车-->"输入当前登录用户的管理员密码"--> ...
- 带你玩转Visual Studio——带你理解微软的预编译头技术
原文地址:http://blog.csdn.net/luoweifu/article/details/49010627 不陌生的stdafx.h 还记得带你玩转Visual Studio——带你新建一 ...
- memory_target
ALTER SYSTEM SET MEMORY_TARGET = 33024M; ALTER SYSTEM SET MEMROY_MAX_TARGET= 33024M SCOPE=SPFILE; sh ...
- JavaScript——封装
封装:使用对象封装的好处是可以减少全局变量污染的机会,讲属性,函数都隶属一个对象. 封装前: <script> var name="foo"; //name是全局的,被 ...
- android手机访问app网页报错:net::ERR_PROXY_CONNECTION_FAILED
手机访问网页报错:net::ERR_PROXY_CONNECTION_FAILED 手机访问app中嵌入的html网页报错: net::ERR_PROXY_CONNECTION_FAILED 原来是手 ...
- 关于Java Web应用中的配置部署描述符web.xml
一.web.xml概述 位于每个Web应用的WEB-INF路径下的web.xml文件被称为配置描述符,这个 web.xml文件对于Java Web应用十分重要,每个Java Web应用都必须包含一个w ...
- git命令行提交并且同步到远程代码库
远程代码库以github为例 1.打开 git bash 2.进入项目目录 cd /e/myGitProjects/test 3.提交到本地git仓库 git add -Agit commit -m ...
- 性能测试二十五:redis-cli 命令总结
常用命令dbsize:查看redis中的kv数量 keys *:查看redis中所有的keyset key_1 v_1:新增一个key_1,包含v_1get key_1:查看key_1中的内容del ...
- myBatis各种依赖包
下载地址:myBatis各种依赖包