链接:
https://www.nowcoder.com/acm/contest/77/B
来源:牛客网
题目描述
给一个数列,会有多次询问,对于每一次询问,会有两种操作:
1:给定两个整数x, y, 然后在原数组的第x位置上加y;
2:给定两个整数l,r,然后输出数组从第l位加到第r位数字的和并换行
输入描述:
第一行有两个整数n, m(1 <= n, m <= 100000)代表数列的长度和询问的次数 第二行n个数字,对于第i个数字a[i],(0<=a[i]<=100000)。 接下来m行,每一行有三个整数f, x, y。第一个整数f是1或者是2,代表操作类型,
如果是1,接下来两个数x,y代表第x的位置上加y,如果是2,则求x到y的和,保证数据合法。
输出描述:
输出每次求和的结果并换行
示例1
输入
10 2 1 2 3 4 5 6 7 8 9 10 1 1 9 2 1 10
输出
64
#include<bits/stdc++.h>
using namespace std;
int main()
{
long long n,m,f,x,y;
long long a[100010];
while(scanf("%lld%lld",&n,&m)!=EOF)
{
int i,j,k=0;
long long sum=0;
a[0]=0;
for(i=1;i<=n;i++)
{
scanf("%lld",&a[i]);
a[i]+=a[i-1]; //缺少 超时 前缀和
}
while(m--)
{
scanf("%lld%lld%lld",&f,&x,&y);
{
if(f==1)
{
for(i=x)
a[x]+=y;
}
else
{
printf("%lld\n",a[y]-a[x-1]);//前缀和解法
}
}
}
}
return 0;
}
一维前缀和
这个优化主要是用来在O(1)时间内求出一个序列a中,a[i]+a[i+1]+……+a[j]的和。
具体原理十分简单:用sum[i]表示(a[1]+a[2]+……+a[i]),其中sum[0]=0,则(a[i]+a[i+1]+……+a[j])即等于sum[j]-sum[i-1]。
二维前缀和
同理,有一维就有二维。对于一个矩阵a,我们也能在O(1)时间内求出子矩阵[x1~x2][y1~y2]的和。
设sum[i][j]为子矩阵[1~i][1~j]的和。则由容斥原理得:
sum[0][j]=sum[i][0]=0
a[x1~x2][y1~y2]=sum[x2][y2]-sum[x1-1][y2]-sum[x2][y1-1]+sum[x1-1][y1-1]
应用问题
核心就两个字:降维。
面对许多高维问题,往往前缀和是最先想到的降维方法。
这样在降维的基础上,许多更进一步的优化才能实现