题目描述

  给你一个长度为\(n\)的数列,还有\(m\)个询问,对于每个询问\((l,r,a,b)\),输出1.区间\([l,r]\)有多少范围在\([a,b]\)的数;2.区间\([l,r]\)有多少范围在\([a,b]\)的权值。

  \(n\leq 100000,m\leq 1000000\)

题解

  这道题莫队可以水过。

  这里讲一个更优秀的算法。

  建一棵权值线段树。每一个点存它代表的范围内所有数的下标。

  一个询问对应权值线段树中的一些点。每个点要求出\([l,r]\)内的数的个数和不同的数的个数。第一问可以乱搞。第二问直接排序后用树状数组维护每个数最后一次出现的位置然后离线乱搞。

  每个点会在根到这个点的路径上各插入一次,所以总的点数是\(n\log n\)。

  总的时间复杂度是\(O((n+m)\log^2 n)\)

代码

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cstdlib>
#include<ctime>
#include<utility>
#include<cmath>
#include<vector>
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int,int> pii;
typedef pair<ll,ll> pll;
void sort(int &a,int &b)
{
if(a>b)
swap(a,b);
}
void open(const char *s)
{
#ifndef ONLINE_JUDGE
char str[100];
sprintf(str,"%s.in",s);
freopen(str,"r",stdin);
sprintf(str,"%s.out",s);
freopen(str,"w",stdout);
#endif
}
struct ques
{
int l,r,a,b,id;
};
vector<ques> q[200010];
vector<pii> c[200010];
struct node
{
int l,r,ls,rs;
};
node a[200010];
int cnt;
int rt;
void build(int &p,int l,int r)
{
p=++cnt;
a[p].l=l;
a[p].r=r;
if(l==r)
return;
int mid=(l+r)>>1;
build(a[p].ls,l,mid);
build(a[p].rs,mid+1,r);
}
void build2(int p)
{
if(a[p].l==a[p].r)
return;
int mid=(a[p].l+a[p].r)>>1;
for(vector<pii>::iterator x=c[p].begin();x!=c[p].end();x++)
if(x->second<=mid)
c[a[p].ls].push_back(*x);
else
c[a[p].rs].push_back(*x);
build2(a[p].ls);
build2(a[p].rs);
}
void add(int &p,ques &v)
{
if(v.a<=a[p].l&&v.b>=a[p].r)
{
q[p].push_back(v);
return;
}
int mid=(a[p].l+a[p].r)>>1;
if(v.a<=mid)
add(a[p].ls,v);
if(v.b>mid)
add(a[p].rs,v);
}
int ans[1000010];
int ans2[1000010];
int d[100010];
int e[100010];
int f[100010];
int n,m;
int bt[100010];
int bt2[100010];
int last[100010];
void add(int x,int v)
{
for(;x<=n;x+=x&-x)
bt[x]+=v;
}
int sum(int x)
{
int s=0;
for(;x;x-=x&-x)
s+=bt[x];
return s;
}
void add2(int x,int v)
{
for(;x<=n;x+=x&-x)
bt2[x]+=v;
}
int sum2(int x)
{
int s=0;
for(;x;x-=x&-x)
s+=bt2[x];
return s;
}
void rd(int &s)
{
int c;
while((c=getchar())<'0'||c>'9');
s=c-'0';
while((c=getchar())>='0'&&c<='9')
s=s*10+c-'0';
}
int cmp(ques a,ques b)
{
return a.r<b.r;
}
int main()
{
open("bzoj3236");
scanf("%d%d",&n,&m);
int i,x;
build(rt,1,n);
for(i=1;i<=n;i++)
{
rd(x);
c[1].push_back(pii(i,x));
}
build2(rt);
ques v;
for(i=1;i<=m;i++)
{
rd(v.l);
rd(v.r);
rd(v.a);
rd(v.b);
v.id=i;
add(rt,v);
}
int j;
for(i=1;i<=cnt;i++)
{
int sz2=q[i].size();
if(!sz2)
continue;
int sz=c[i].size();
int k=1;
for(j=1;j<=sz;j++)
{
d[j]=c[i][j-1].first;
f[j]=c[i][j-1].second;
}
sort(q[i].begin(),q[i].end(),cmp);
while(k<=sz2&&q[i][k-1].r<d[1])
k++;
for(j=1;j<=sz;j++)
{
e[j]=last[f[j]];
last[f[j]]=j;
if(e[j])
add(e[j],-1);
add(j,1);
add2(j,1);
while(k<=sz2&&(j==sz||q[i][k-1].r<d[j+1]))
{
int l=lower_bound(d+1,d+sz+1,q[i][k-1].l)-d;
ans[q[i][k-1].id]+=sum(j)-sum(l-1);
ans2[q[i][k-1].id]+=sum2(j)-sum2(l-1);
k++;
}
}
for(j=1;j<=sz;j++)
{
if(last[f[j]])
add(last[f[j]],-1);
last[f[j]]=0;
add2(j,-1);
}
}
for(i=1;i<=m;i++)
printf("%d %d\n",ans2[i],ans[i]);
return 0;
}

【BZOJ3236】【AHOI2013】作业 线段树 分治 树状数组的更多相关文章

  1. BZOJ3236 [Ahoi2013]作业 【莫队 + 树状数组】

    题目链接 BZOJ3236 题解 没想到这题真的是如此暴力 #include<algorithm> #include<iostream> #include<cstring ...

  2. [bzoj3809]Gty的二逼妹子序列/[bzoj3236][Ahoi2013]作业

    [bzoj3809]Gty的二逼妹子序列/[bzoj3236][Ahoi2013]作业 bzoj   bzoj 题目大意:一个序列,m个询问在$[l,r]$区间的$[x,y]$范围内的数的个数/种类. ...

  3. 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 ...

  4. [BZOJ3236][AHOI2013]作业:树套树/莫队+分块

    分析 第一问随便搞,直接说第二问. 令原数列为\(seq\),\(pre_i\)为\(seq_i\)这个值上一个出现的位置,于是可以简化询问条件为: \(l \leq i \leq r\) \(a \ ...

  5. [BZOJ3236]:[Ahoi2013]作业(莫队+分块)

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

  6. 树分治&树链剖分相关题目讨论

    预备知识 树分治,树链剖分   poj1741 •一棵有n个节点的树,节点之间的边有长度.方方方想知道,有多少个点对距离不超过m 题解 点分治模板题.详见我早上写的http://www.cnblogs ...

  7. BZOJ3236:[AHOI2013]作业(莫队,分块)

    Description Input Output Sample Input 3 4 1 2 2 1 2 1 3 1 2 1 1 1 3 1 3 2 3 2 3 Sample Output 2 2 1 ...

  8. 【莫队算法】【权值分块】bzoj3236 [Ahoi2013]作业

    莫队显然.然后维护转移的时候如果用树状数组,则很容易TLE.所以用权值分块维护转移. 总复杂度O(m*sqrt(n)). #include<cstdio> #include<algo ...

  9. bzoj3809 Gty的二逼妹子序列 & bzoj3236 [Ahoi2013]作业 莫队+分块

    题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=3809 https://lydsy.com/JudgeOnline/problem.php?id ...

随机推荐

  1. Python是如何进行内存管理

    三个方面:一对象的引用计数机制,二垃圾回收机制,三内存池机制 一.对象的引用计数机制 Python内部使用引用计数,来保持追踪内存中的对象,所有对象都有引用计数. 引用计数增加的情况: 1,一个对象分 ...

  2. 了解sso原理

  3. fun = [lambda x: x*i for i in range(4)] 本质解析/原理,LEGB规则 闭包原理

    命名空间,闭包原理,参考点击本文 一.问题描述 fun = [lambda x: x*i for i in range(4)] for item in fun: print(item(1)) 上述式子 ...

  4. Socket编程,SocketServer模块

    一.SocketServer的几种类型 面向远程: TCP 协议链接:socketserver.TCPServer(server_address, RequestHandlerClass, bind_ ...

  5. Elasticsearch 5.0Head插件

    Elasticsearch 5.0 —— Head插件部署指南   使用ES的基本都会使用过head,但是版本升级到5.0后,head插件就不好使了.下面就看看如何在5.0中启动Head插件吧! 官方 ...

  6. Django之在Python中调用Django环境

    Django之在Python中调用Django环境 新建一个py文件,在其中写下如下代码: import os if __name__ == '__main__': os.environ.setdef ...

  7. 课程存储校对:程序设计思想、源程序代码、运行结果截图,以及开发过程中的项目计划日志、时间记录日志、缺陷记录日志(PSP0级记录)。

    1.程序设计思想 ⑴将JDBC驱动jar包导入到WEB-INF的lib文件夹下 ⑵建立数据库,在数据库中建表,分别将课程名称.任课教师及上课地点录入到列中 ⑶首先写出加载驱动.关闭资源的工具类和异常处 ...

  8. bat 文本合并

    小工具—把多个TXT文件合成一个 - TTXS_RS的博客 - CSDN博客https://blog.csdn.net/TTXS_RS/article/details/79743384 把所有文本文件 ...

  9. Cannot connect to database because the database client

    问题描述: arcgis server10.1  arcgis sde10出现下面问题 Cannot connect to  database because the database client ...

  10. java不同的包下相同的类名的问题与解决办法

    Java中的类以包进行分类组织,当程序中需要用到某个包下的类时,可以以该类的全限定名进行引用.这样,不同的包中的类就可以同名,不会产生混淆. 但是这样就可能导致引用的时候会产生一些问题. 第一个问题, ...