题目链接: http://acm.zzuli.edu.cn/problem.php?id=1531
1531: 小L的区间求和
时间限制: 1 Sec 内存限制: 128 MB
题目描述
在给定的一个整数序列中,小L希望找到一个连续的区间,这个区间的和能够被k整除,请你帮小L算一下满足条件的最长的区间长度是多少。
输入
第一行输入两个整数n、k。(1 <= n <= 105,1<=k<100)
接下来一行输入n个整数,表示序列中的数。
输出
输出一个整数,满足条件区间的最长长度,如果不存在,输出0
样例输入
5 7 1 2 4 1 1
样例输出
3
思路:先看代码:
#include <bits/stdc++.h>
using namespace std;
int a[100005], s[100005];
map<int, int>M;
int main()
{
int n, k;
scanf("%d%d", &n, &k);
for(int i=1; i<=n; i++)
{
scanf("%d", &a[i]);
s[i] = (s[i-1]+a[i])%k;//s[i]保存的是从第一个数到当前数对7余值的和,出现和为7则s[i]的值会和之前某s[i]重复
M[s[i]] = -1;
}
int ans = 0;
M[0] =0 ;//在下面的for循环中,针对每个s[i]的值,m[s[i]]仅对应一个特殊值,因题目数据问题,将m[0]设置成0或任何大于7的值即可,但对于1 2 4 这种输入,设置成0才能得到正确输出。
for(int i=1; i<=n; i++)
{
if(M[s[i]] == -1)//如果m[s[[i]]!=-1,则说明有7出现
{
M[s[i]] = i;
}
else
{
ans = max(ans, i-M[s[i]]);//有七出现时,m[s[i]]对应的是上次出现相同余值时位置
}
}
printf("%d\n", ans);
return 0;
}
m[s[i]] | 1 | 2 | 3 | 4 | 2 | 3 | 7 | 2 | 2 |
---|---|---|---|---|---|---|---|---|---|
s[i] | 1 | 2 | 3 | 5 | 2 | 3 | 4 | 2 | 2 |
对应输入 | 1 | 1 | 1 | 2 | 4 | 1 | 1 | 5 | 7 |
i | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 |
如图:s[i]出现重复则用当前位置(i)减去上次s[i]的值出现的位置(m[s[i]]),即得到距离。