### 洛谷P4396  题目链接 ###

题目大意:

有 n 个整数组成的数组,m 次询问,每次询问中有四个参数 l ,r,a,b 。问你在[l,r] 的区间内的所有数中,值属于[a,b] 的数的个数以及种类数。

分析:

1、由于可以离线操作,故采用莫队。

2、由于在莫队的基础上还涉及区间[a,b]的值的个数,故可以用前缀和的思想,求得出sum(b) - sum(a - 1)即可。由于与莫队使用是动态的,故需要用树状数组维护,因为可以 logn 动态插入。

3、对于求区间种类数,需要用第二个树状数组维护。且需要用 cnt[] 数组来标记当前数是否是第一次出现或最后一次出现的数。如果是第一次出现且需 add,则更新当前点以及后置点 + +(树状数组插入);若为最后一次出现且需 del,则更新当前点以及后置点 - - 即可。

4、由于数组中数的范围未给定,在树状数组中可能爆空间,故需要离散化。

算法正确性:

树状数组的原理在于,若当前点值为 x ,则对于所有>= x 的数,都要加上这个数的贡献,即 + + 。比如有 >= x 的数 y ,在求前缀和(即在求 <= y 的数的个数)时,所有出现过的且 <= y 的值的点 x ,都在之前的插入对答案做有贡献。可想而知,树状数组的这种优点导致可以存储前缀和。

然后再注意一下离散化的细节即可,最好在去重的末端加入一个极大值,这样lower_bound 就不会越界,且在求 pos1 以及 pos2 时能很好地判断取值。

然后莫队排序时,需要用到奇偶排序,不然会 T 一个点。

代码如下:

#include<iostream>
#include<algorithm>
#include<string.h>
#include<cmath>
using namespace std;
#define inf 0x3f3f3f3f
#define maxn 100008
int n,m,block,len;
int be[maxn];
int g[maxn],f[maxn],d[maxn];
int c[maxn],z[maxn],cnt[maxn];
struct S{
int ans1,ans2;
}s[maxn];
struct Mo{
int id;
int l,r;
int a,b;
}A[maxn];
bool cmp(Mo q,Mo w){
return (be[q.l]^be[w.l])?q.l<w.l:(be[q.l]&)?q.r<w.r:q.r>w.r; //奇偶排序
}
inline int lowbit(int x){return x&(-x);}
void Zupdate(int i,int q){
while(i<=len){
z[i]+=q;
i+=lowbit(i);
}
return;
}
int Zquery(int i){
int ans=;
while(i){
ans+=z[i];
i-=lowbit(i);
}
return ans;
}
void update(int i,int q){
while(i<=len){
c[i]+=q;
i+=lowbit(i);
}
return;
}
inline int query(int i){
int ans=;
while(i){
ans+=c[i];
i-=lowbit(i);
}
return ans;
}
void add(int x){
int pos=d[x];
update(pos,);
if(!cnt[pos]) Zupdate(pos,);
cnt[pos]++;
return;
}
void del(int x){
int pos=d[x];
update(pos,-);
cnt[pos]--;
if(!cnt[pos]) Zupdate(pos,-);
return;
}
int main()
{
scanf("%d%d",&n,&m);
block=sqrt(n);
for(int i=;i<=n;i++) {
scanf("%d",&g[i]);
be[i]=(i-)/block+;//分块
f[i]=g[i];
}
for(int i=;i<=m;i++){
scanf("%d%d%d%d",&A[i].l,&A[i].r,&A[i].a,&A[i].b);
A[i].id=i;
}
sort(A+,A+m+,cmp);
sort(f+,f+n+);
len=unique(f+,f+n+)-f-;
f[len+]=inf;
for(int i=;i<=n;i++) d[i]=lower_bound(f+,f+len+,g[i])-f;// 原数组g[i]中的离散值 d[i]
int l=,r=;
for(int i=;i<=m;i++){
while(l<A[i].l) del(l++);
while(l>A[i].l) add(--l);
while(r<A[i].r) add(++r);
while(r>A[i].r) del(r--);
int pos1=lower_bound(f+,f+len+,A[i].a)-f;
int pos2=lower_bound(f+,f+len+,A[i].b)-f;
if(f[pos2]>A[i].b) pos2--;
s[A[i].id].ans1=query(pos2)-query(pos1-);
s[A[i].id].ans2=Zquery(pos2)-Zquery(pos1-);
}
for(int i=;i<=m;i++) printf("%d %d\n",s[i].ans1,s[i].ans2 );
}

洛谷 P4396 (离散化+莫队+树状数组)的更多相关文章

  1. 【BZOJ3460】Jc的宿舍(树上莫队+树状数组)

    点此看题面 大致题意: 一棵树,每个节点有一个人,他打水需要\(T_i\)的时间,每次询问两点之间所有人去打水的最小等待时间. 伪·强制在线 这题看似强制在线,但实际上,\(pre\ mod\ 2\) ...

  2. bzoj3236 作业 莫队+树状数组

    莫队+树状数组 #include<cstdio> #include<cstring> #include<iostream> #include<algorith ...

  3. BZOJ_3289_Mato的文件管理_莫队+树状数组

    BZOJ_3289_Mato的文件管理_莫队+树状数组 Description Mato同学从各路神犇以各种方式(你们懂的)收集了许多资料,这些资料一共有n份,每份有一个大小和一个编号 .为了防止他人 ...

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

  5. COGS.1822.[AHOI2013]作业(莫队 树状数组/分块)

    题目链接: COGS.BZOJ3236 Upd: 树状数组实现的是单点加 区间求和,采用值域分块可以\(O(1)\)修改\(O(sqrt(n))\)查询.同BZOJ3809. 莫队为\(O(n^{1. ...

  6. bzoj 3289: Mato的文件管理 莫队+树状数组

    3289: Mato的文件管理 Time Limit: 40 Sec  Memory Limit: 128 MB[Submit][Status][Discuss] Description Mato同学 ...

  7. 51nod 1290 Counting Diff Pairs | 莫队 树状数组

    51nod 1290 Counting Diff Pairs | 莫队 树状数组 题面 一个长度为N的正整数数组A,给出一个数K以及Q个查询,每个查询包含2个数l和r,对于每个查询输出从A[i]到A[ ...

  8. HihoCoder 1488 : 排队接水(莫队+树状数组)

    描述 有n个小朋友需要接水,其中第i个小朋友接水需要ai分钟. 由于水龙头有限,小Hi需要知道如果为第l个到第r个小朋友分配一个水龙头,如何安排他们的接水顺序才能使得他们等待加接水的时间总和最小. 小 ...

  9. BZOJ 3236 莫队+树状数组

    思路: 莫队+树状数组 (据说此题卡常数) yzy写了一天(偷笑) 复杂度有点儿爆炸 O(msqrt(n)logn) //By SiriusRen #include <cmath> #in ...

随机推荐

  1. django----多对多三种创建方式 form组件

    目录 多对多三种创建方式 全自动 全手动 半自动 form组件 基本使用 form_obj 及 is_valid() 前端渲染方式 取消前端自动校验 正则校验 钩子函数(Hook方法) cleaned ...

  2. Linux常见指令大全

    转载自https://www.cnblogs.com/caozy/p/9261224.html 前言 本文特点 授之以渔:了解命令学习方法.用途:不再死记硬背,拒绝漫无目的: 准确无误:所有命令执行通 ...

  3. .NET Core环境变量和用户秘钥实现开发中的数据安全

    目录 一.注入 IConfiguration 二.从配置文件 appsettings.json 中获取环境变量 三.从项目中获取环境变量 四.用户秘钥设置环境变量 前言:有很多人将秘钥,数据库连接字符 ...

  4. 2.java三大特性

    1. 封装 方法:将属性值修饰为私有,提供get和set方法.造成所有对对象的访问都是通过方法的调用来完成(配合this的使用) 结果:用户不能直接随意改变一个对象内的属性,必须通过调用方法(验证)来 ...

  5. 《Java基础知识》Java final关键字:阻止继承和多态

    在 Java 中,声明类.变量和方法时,可使用关键字 final 来修饰.final 所修饰的数据具有“终态”的特征,表示“最终的”意思.具体规定如下: final 修饰的类不能被继承. final ...

  6. 从KafkaConsumer看看Kafka(一)

      Kafka的消息模型为发布订阅模型,消息生产者将消息发布到主题(topic)中,一个或多个消费者订阅(消费)该主题消息并消费,此模型中发布到topic中的消息会被所有消费者所订阅到,先介绍Kafk ...

  7. c++-继承的学习

    继承的基本概念 继承和派生 继承概念 派生类的访问控制(继承三种方式.类三种访问控制.三看原则)综合训练 继承中的构造和析构 类型兼容性原则 继承中的构造和析构 继承中同名成员函数.成员变量处理方法 ...

  8. 基于Tomcat的GeoServer部署步骤

    一.安装JAVA 资源:JDK1.8 提取码:0y26 步骤: 1.安装完成后,右击"我的电脑",点击"属性",选择"高级系统设置": 2. ...

  9. Jetpack架构组件(二)Lifecycle使用

    1.直接添加如下依赖就可以满足日常的工作,如果缺少哪个库,再去单独添加就好了 implementation "android.arch.lifecycle:extensions:1.1.1& ...

  10. C/C++ 项目编译工具简介

    本文基于一个需要对 C 语言家族项目进行编译.生成解决方案的开发者的视角,对编译过程中所需要的各个工具进行简要的名词解释. GCC | LLVM 提供编译器 <-- GNU Make 根据配置文 ...