K倍区间 刷题笔记

法一 前缀和暴力搜索 (数据大会超时)

#include<iostream>

#include<cstring>

#include<algorithm>

#include<cstdio>

using namespace std;

const int N=100010;

int aN,sN;

int n,k;

int main(){

cin>>n>>k;

for(int i=1;i<=n;i++){

cin>>ai;

si=si-1+ai;

}

int cnt=0;

for(int i=1;i<=n;i++){

for(int j=1;j<=i;j++){

int ans=si-sj-1;

if(ans%k==0){

cnt++;

}

}

}

cout<<cnt;

return 0;

}`

法二 根据 式子推导

if((si-sj-1)%k==0)答案++;

换句话说 si和sj-1对k取模的余数是一样的

此处 拿样例

5 2

1 2 3 4 5

可以看到s1,s2,s5,对k取模的余数是一样的

且s0%k=0;

s2和s1,s5和s1,s5和s2能构成三个k倍区间

s3和s0,s4和s0,s4和s3能构成三个k倍区间所以一共六个K倍区间

因此

我们引出 开一个cnt数组来记录该余数的出现次数

例如 余数2出现了两次 说明当前已经有两个sl-1对k取模余数为2

后面一旦出现当前余数2 就可以与这两个sl-1形成k倍区间

所以 答案+上目前余数出现的次数

然后此时余数2已经出现三次 则让cnt2++;即cnt2=3;

将其标记为出现三次 后面一旦再出出现余数2的sr

答案直接+3 同时将余数2 标记成四次

代码

#include<iostream>

#include<cstring>

#include<algorithm>

#include<cstdio>

using namespace std;

typedef long long ll;

const int N=100010;

ll aN,sN,cntN;

ll n,k;

int main(){

cin>>n>>k;

for(int i=1;i<=n;i++){

cin>>ai;

si=si-1+ai;

}

ll res=0;

cnt0=1;

for(int i=1;i<=n;i++){

res+=cnts\[i%k];

cnts\[i%k]++;

}

cout<<res;

return 0;

}

相关推荐
whyTeaFo2 小时前
MIT 6.1810: Lec 5: calling conventions and stack frames RISC-V
笔记
上课不要睡觉了3 小时前
【统计法规】4.1统计管理体制概述
笔记·统计师考试
墨白曦煜3 小时前
算法实战笔记:剥开回溯算法的外衣——从通用模板到高阶去重(八)
笔记·算法
Upsy-Daisy4 小时前
IOTA 学习笔记(四):当前 IOTA 架构总览
笔记·学习·架构
山楂树の4 小时前
JS中??和||的区别
笔记
U盘失踪了4 小时前
Browser Use — AI驱动浏览器自动化的全新范式
笔记·自动化
疯狂打码的少年5 小时前
Cache的三种映射方式(直接/全相联/组相联)
linux·服务器·数据库·笔记
RainCity5 小时前
Java Swing 自定义组件库分享(十)
java·笔记·后端
FakeOccupational6 小时前
【数学 代数学】无理数:\sqrt2和\pi 是无理数的证明 + 无理数集合的“非正则”性质(暂记)
笔记
是小王同学啊~6 小时前
Kafka 面试通关笔记:高频八股 + 生产实战 + 追问链路(上)
笔记·面试·kafka