~~~题面~~~

题解:

  题目要求统计一个区间内数值在[a, b]内的数的个数和种数,而这个是可以用树状数组统计出来的,所以可以考虑莫队。

  考虑区间[l, r]转移到[l, r + 1],那么对于维护个数的树状数组就直接加即可。

  对于维护种数的树状数组,我们额外维护一个数组num,表示数a在区间内出现了多少次,如果是新出现的,那么就加入树状数组。
  如果要删除一个数并且这个数在区间内只出现了一次,那么就删除这个数。注意不论什么情况都要实时维护num数组。

  然后莫队即可。

  Gty的二逼妹子序列是洛谷P4867和作业的某一问是一模一样的,把数组开大点就可以过。

 #include<bits/stdc++.h>
using namespace std;
#define R register int
#define AC 100100 int n, m, cnt, block;
int ans1[AC], ans2[AC], s[AC], num[AC], tot[AC]; struct node{
int l, r, a, b, id;
}q[AC]; inline int lowbit(int x)
{
return x & (-x);
} struct kkk{
int a[AC]; void add(int x, int y)
{
for(R i = x; i <= cnt; i += lowbit(i)) a[i] += y;
} int find(int x)
{
int rnt = ;
for(R i = x; i; i -= lowbit(i)) rnt += a[i];
return rnt;
}
}c1, c2; inline int read()
{
int x = ;char c = getchar();
while(c > '' || c < '') c = getchar();
while(c >= '' && c <= '') x = x * + c - '', c = getchar();
return x;
} int half(int x)//查询离散化后的值
{
int l = , r = cnt, mid;
while(l < r)
{
mid = (l + r) >> ;
if(num[mid] == x) return mid;
else if(num[mid] > x) r = mid;
else l = mid + ;
}
return l;
} int half1(int x)//查询最小的大于等于a的值
{
int l = , r = cnt, mid;
if(num[cnt] < x) return cnt + ;
while(l < r)
{
mid = (l + r) >> ;
if(num[mid] >= x) r = mid;
else l = mid + ;
}
return l;
} int half2(int x)//查询最大的小于等于b的值
{
int l = , r = cnt, mid;
if(num[] > x) return ;
while(l < r)
{
mid = (l + r + ) >> ;//强制偏右
if(num[mid] > x) r = mid - ;
else l = mid;
}
return l;
} inline bool cmp(node a, node b)
{
if(a.l / block != b.l / block) return a.l < b.l;
else return a.r < b.r;//分块排序
} void pre()
{
n = read(), m = read(), block = sqrt(n);
for(R i = ; i <= n; i ++) s[i] = num[i] = read();
sort(num + , num + n + );
for(R i = ; i <= n; i ++)
if(num[i] != num[i + ]) num[++cnt] = num[i];
for(R i = ; i <= n; i ++) s[i] = half(s[i]);//在这里离散化,这样后面就不用调用了
for(R i = ; i <= m; i ++)
{
q[i].l = read(), q[i].r = read(), q[i].id = i;
q[i].a = half1(read()), q[i].b = half2(read());
}
sort(q + , q + m + , cmp);
} void add(int x)
{
c1.add(x, );
if(!tot[x]) c2.add(x, );
++ tot[x];
} void del(int x)
{
c1.add(x, -);
-- tot[x];
if(!tot[x]) c2.add(x, -);
} void work()//这里每个点的贡献与区间无关,相对独立,所以不用考虑顺序问题
{
int l, r;
for(R i = q[].l; i <= q[].r; i ++) add(s[i]);
ans1[q[].id] = c1.find(q[].b) - c1.find(q[].a - );
ans2[q[].id] = c2.find(q[].b) - c2.find(q[].a - );
l = q[].l, r = q[].r;
for(R i = ; i <= m; i ++)
{
int ll = q[i].l, rr = q[i].r;
while(ll < l) -- l, add(s[l]);
while(ll > l) del(s[l]), ++ l;
while(rr > r) ++ r, add(s[r]);
while(rr < r) del(s[r]), -- r;
ans1[q[i].id] = c1.find(q[i].b) - c1.find(q[i].a - );
ans2[q[i].id] = c2.find(q[i].b) - c2.find(q[i].a - );
}
for(R i = ; i <= m; i ++) printf("%d %d\n", ans1[i], ans2[i]);
} int main()
{
// freopen("in.in", "r", stdin);
pre();
work();
// fclose(stdin);
return ;
}

[AHOI2013]作业 & Gty的二逼妹子序列 莫队的更多相关文章

  1. Bzoj 3809: Gty的二逼妹子序列 莫队,分块

    3809: Gty的二逼妹子序列 Time Limit: 35 Sec  Memory Limit: 28 MBSubmit: 868  Solved: 234[Submit][Status][Dis ...

  2. BZOJ 3809 Gty的二逼妹子序列 莫队算法+分块

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

  3. [BZOJ3809]Gty的二逼妹子序列[莫队+分块]

    题意 给出长度为 \(n\) 的序列,\(m\) 次询问,每次给出 \(l,r,a,b\) ,表示询问区间 \([l,r]\) 中,权值在 \([a,b]\) 范围的数的种类数. \(n\leq 10 ...

  4. 【BZOJ3809】Gty的二逼妹子序列 莫队 分块

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

  5. bzoj 3809 Gty的二逼妹子序列——莫队+分块

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3809 容易想到树状数组维护值域.但修改和查询都是 log 太慢. 考虑有 nsqrt(n) ...

  6. bzoj 3809 Gty的二逼妹子序列 —— 莫队+分块

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3809 据说一开始应该想到莫队+树状数组,然而我想的却是莫队+权值线段树... 如果用权值线段 ...

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

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

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

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

  9. 3809: Gty的二逼妹子序列

    3809: Gty的二逼妹子序列 链接 分析: 和这道AHOI2013 作业差不多.权值是1~n的,所以对权值进行分块.$O(1)$修改,$O(\sqrt n)$查询. 代码: #include< ...

随机推荐

  1. __name__ 和 "__main__"

    本模块名: person 调用者模块名: start import sys def funcperson(): print('我是人') print(sys.modules[__name__]) # ...

  2. php Trait的使用

    1.php中的trait是啥? 看上去既像类又像接口,其实都不是,Trait可以看做类的部分实现,可以混入一个或多个现有的PHP类中,其作用有两个:表明类可以做什么:提供模块化实现.Trait是一种代 ...

  3. Mysql 5.7 开启远程连接

    1 在控制台执行 mysql -uroot -p 系统提示输入数据库root用户的密码,输入完成后即进入mysql控制台 2 选择数据库 mysql -uroot -p use mysql; 开启远程 ...

  4. flask的模板

    flask用的是jinja2的模板 模板其实是一个包含响应文本的文件,其中用占位符(变量)表示动态部分,告诉模板引擎其具体的值需要从使用的数据中获取 使用真实值替换变量,再返回最终得到的字符串,这个过 ...

  5. zabbix配置报警媒介-用户-动作-邮件脚本触发mailx邮件报警

    2018-09-16更新,新版本zabbix不需要使用脚本发送邮件,在zabbix web界面直接配置就可以 配置邮件参数,测试发送邮件 确认安装相关服务,centos7默认安装 [root@VM_1 ...

  6. java入门---基础语法&基础常识&编码规范&命名规范

        一个Java程序可以认为是一系列对象的集合,而这些对象通过调用彼此的方法来协同工作.下面简要介绍下类.对象.方法和实例变量的概念. 对象:对象是类的一个实例,有状态和行为.例如,一条狗是一个对 ...

  7. [Python 3.X]python练习笔记[2]-----用python实现七段数码管显示年月日

    #SevenDigitsDrawV2.py import turtle import time def drawGap(i):#绘制数码管间隔 turtle.penup() turtle.fd(i) ...

  8. 虚拟现实-VR-UE4-编译UE4源码

    通过Git将UE4源代码获取到本地计算机 切记路径不要有中文 这里面我已经在进行编译了,有部分文件是多余出来的, 第一步就是点击 setup.bat批处理,这个过程回取决与你的网速的快慢,我等了一下午 ...

  9. TortoiseSVN的安装使用

    下面分享一篇关于TortoiseSVN的安装以及使用 1.运行TortoiseSVN-1.6.6.17493-win32-svn-1.6.6.msi程序, 开始安装 2.点击Next, 下一步 3.选 ...

  10. CentOS Linux release 7.5.1804下安装MySQL5.7.24

    1.环境查看: 2.卸载自带MariaDB数据库: 3.下载MySQL5.7.14安装包: 4.使用wget工具下载需要安装数据库的依赖包: 5.解压缩bundel包: 6.按照顺序进行安装: 7.数 ...