Yura
Portal --> broken qwq
Description
给你一个长度为\(n\)的序列\(a\)和一个正整数\(k\),求满足如下条件的区间\([l,r]\)的数量:\((\sum\limits_{i=l}^r a[i]-max(l,r))\%k=0\),其中\(max\)表示的是该区间的最大值
数据范围:\(,n<=300000,k<=100000\),保证答案\(<=10^9\)
Solution
啊所以说是。。想不到怎么做就想分治吗qwq
我们首先可以考虑一种最简单粗暴的求解方式:我们固定一个左端点和最大值,然后看右端点能够移到哪里,然后统计其贡献,快速得出一个区间的表达式值是比较方便的,只要直接维护一个前缀和\(sum\),\(sum[r]-sum[l-1]-mx\)就是表达式的值了
然后怎么优化的话。。就是可以将这个思路用到分治里面去
对于当前的区间\([l,r]\),我们考虑左端点在\([l,mid]\),右端点在\([mid+1,r]\)中的满足条件的区间, 具体的计算有点迷,考虑将这堆区间再按照最大值的位置分成两类,一类是最大值在\(mid\)及以前,一类是最大值在\(mid\)以后,这两类的计算方式类似,接下来以第一类为例说一下大致过程:一开始将左端点设在\(mid\)处,将右端点设在\(mid+1\)处,每次将左端点往左移一位,然后更新当前位置到\(mid\)这段的最大值作为整个区间的最大值,然后暴力将右端点往右移,这里我们需要一个计数数组来统计贡献,考虑到我们要算的是\(sum[r]-sum[l-1]-mx\equiv 0(mod\ k)\)的区间,而当前\(l\)和\(mx\)是固定的,所以我们可以将右端点的贡献记在计数数组中\(sum[r]\%k\)的位置,调用的时候直接调\((sum[l-1]+mx)\%k\)位置的值就是贡献了
如果说是统计第二类的话,就是变成将贡献记在\(sum[l-1]\%k\)的位置上,调用的时候是\((sum[l-1]-mx)\%k\)位置的值即可
最后是。。记得清空
还有就是相同的情况只能算到一边去,也就是其中一类在移的时候是\(<=mx\),另一类是\(<mx\),否则的话会算重
代码大概长这个样子
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
const int N=300010;
int a[N],sum[N],cnt[N];
int n,m,ans;
void solve(int l,int r){
if (l==r) return;
int mid=l+r>>1,tpl,tpr,mx;
tpl=mid; tpr=mid+1; mx=0;
while (tpl>=l){
mx=max(mx,a[tpl]);
while (tpr<=r&&mx>=a[tpr])
++cnt[sum[tpr]],++tpr;
ans+=cnt[(sum[tpl-1]+mx)%m];
--tpl;
}
for (int i=mid+1;i<=r;++i) cnt[sum[i]]=0;
tpl=mid; tpr=mid+1; mx=0;
while (tpr<=r){
mx=max(mx,a[tpr]);
while (tpl>=l&&mx>a[tpl])
++cnt[sum[tpl-1]],--tpl;
ans+=cnt[(m-mx%m+sum[tpr])%m];
++tpr;
}
for (int i=l;i<=mid;++i) cnt[sum[i-1]]=0;
solve(l,mid);
solve(mid+1,r);
}
int main(){
#ifndef ONLINE_JUDGE
freopen("a.in","r",stdin);
#endif
scanf("%d%d",&n,&m);
sum[0]=0;
for (int i=1;i<=n;++i) scanf("%d",a+i),sum[i]=(sum[i-1]+a[i])%m;
solve(1,n);
printf("%d\n",ans);
}
Yura的更多相关文章
- Looksery Cup 2015 F - Yura and Developers 单调栈+启发式合并
F - Yura and Developers 第一次知道单调栈搞出来的区间也能启发式合并... 你把它想想成一个树的形式, 可以发现确实可以启发式合并. #include<bits/stdc+ ...
- codeforces 549F Yura and Developers(分治、启发式合并)
codeforces 549F Yura and Developers 题意 给定一个数组,问有多少区间满足:去掉最大值之后,和是k的倍数. 题解 分治,对于一个区间,找出最大值之后,分成两个区间. ...
- 【Codeforces549F】Yura and Developers [单调栈][二分]
Yura and Developers Time Limit: 20 Sec Memory Limit: 512 MB Description Input Output Sample Input 4 ...
- Codeforces 549F Yura and Developers
probelm 题意 给定一个序列和一个mod值,定义[l,r]合法当l到r的全部元素和减去当中的最大值的结果能够整除mod.问共同拥有多少区间合法. 思路 一開始想的分治. 对于一个[l,r]我们能 ...
- ●CodeForces 549F Yura and Developers
题链: http://codeforces.com/problemset/problem/549/F题解: 分治,链表. 考虑对于一个区间[L,R],其最大值在p位置, 那么答案的贡献就可以分为3部分 ...
- Swing做的非阻塞式仿飞秋聊天程序
采用Swing 布局 NIO非阻塞式仿飞秋聊天程序, 切换皮肤颜色什么的小功能以后慢慢做 启动主程序. 当用户打开主程序后自动获取局域网段IP可以在 设置 --> IP网段过滤, 拥有 JMF ...
- 从零开始学习jQuery (九) jQuery工具函数
一.摘要 本系列文章将带您进入jQuery的精彩世界, 其中有很多作者具体的使用经验和解决方案, 即使你会使用jQuery也能在阅读中发现些许秘籍. 我们经常要使用脚本处理各种业务逻辑, 最常见的就 ...
- 【原创】Android 对话框的使用
对话框即Dialog .google的官方解释:A dialog is usually a small window that appears in front of the current Acti ...
- Yandex 2013Q(Atoms: There and Back Again-贪心+模拟+List)
Atoms: There and Back Again Time limit 2 seconds Memory limit 256Mb Input stdin Output stdout Legend ...
随机推荐
- Java+Selenium 3.x 实现Web自动化 - Maven打包TestNG,利用jenkins执行测试
1. Jenkins本地执行测试 or 服务器端执行测试 测试代码计划通过jenkins执行时,通过网上查询各种教程,大多数为本地执行测试,由此可见,本地执行是大多数人的选择. 经过探讨,最终决定采用 ...
- sql server 查询所有被锁表并批量解除
废话不多说,直接上代码: --查询被锁表 select request_session_id spid,OBJECT_NAME(resource_associated_entity_id) table ...
- 人脸检测及识别python实现系列(6)——终篇:从实时视频流识别出“我”
人脸检测及识别python实现系列(6)——终篇:从实时视频流识别出“我” 终于到了最后一步,激动时刻就要来临了,先平复一下心情,把剩下的代码加上,首先是为Model类增加一个预测函数: #识别人脸 ...
- C# 如何使用 RabbitMQ 实现消息收发
本文是基于http://www.cnblogs.com/cheng-lei/articles/7274513.html的项目结构进行搭建的,了解之前请先阅读http://www.cnblogs.com ...
- Linux内核学习笔记(4)-- wait、waitpid、wait3 和 wait4
进程调用 exit() 退出执行后,被设置为僵死状态,这时父进程可以通过 wait4() 系统调用查询子进程是否终结,之后再进行最后的操作,彻底删除进程所占用的内存资源. wait4() 系统调用由 ...
- 电脑提示‘您需要来自Administration的权限才能对此文件夹进行更改’怎么删除文件
电脑提示'您需要来自Administration的权限才能对此文件夹进行更改'怎么删除文件 应该怎么做 win7系统需要定期删除一些无用的文件,扩大内存空间,但是在删除文件的时候弹出提示"您 ...
- 基于Docker Compose构建的MySQL MHA集群
Docker MySQL MHA 基于Docker 1.13.1之上构建的MySQL MHA Docker Compose Project 可快速启动GTID模式下的MasterHA集群, 主用于My ...
- kafka浅谈
关键词 producer 生产者 broker 缓存代理 consumer 消费者 partition 分区 topic 主题 ...
- Python3【基础】-表达式与运算符
一.什么是表达式? 1+2*3就是一个表达式,这里的加号和乘号叫做运算符,1.2.3叫做操作数.1+2*3计算的结果是7,计算结果可以存到一个变量中,即:res = 1 + 2 * 3. 所谓的表达式 ...
- 关于cnblog.com的用户体验
首先我自己目前是一个学生党,每天在博客园上就上发布一些自己做的东西以及老师布置的作业,还能在上面学习很多别人的一些好的列子,我就希望博客园能够很好地为我们这些学生服务,当我们用它时能够很好地达到我们的 ...