前缀和
一维:通过空间换时间适用于需要频繁查询某一段区间和的场景。
一维数组:
一维前缀和中的每一项:
,该前缀和中的每一项也就是数组中对应的前 i 项和。
一维前缀和数组的构造:
在输入原数组时同步构造前缀和数组可以改写为
cpp
for(int i=1;i<=n;i++){
scanf("%d",&arr[i]);
crr[i]=crr[i-1]+arr[i];
}
通常前缀和数组从下标为1开始的值默认为0
一维区间和查询:
查询原数组区间为 [ l , r ] 的元素的和:
二维:通过空间换时间适用于需要频繁查询某一个子矩阵中元素和的场景。
二维数组:
. . . . . . . . . . . . . . .
二维前缀和中的每一项:
,该前缀和中的每一项也就是数组中以和这两个元素为对角线构成的矩形中元素和。
二维前缀和数组的构造:
也是在输入原数组时同步构造前缀和数组 根据二维数组一行一行地读入顺序可以改写为 ,运用这种类似于dp的想法就可以实现。
cpp
for(int i=1;i<=n;i++){
int sum=0;
for(int j=1;j<=m;j++){
scanf("%d",drr[i][j]);
sum+=drr[i][j];
crr[i][j]=crr[i-1][j]+sum;
}
}
和一维类似:行和列的都从1开始
二维子矩阵元素和查询:
查询以和为对角线构成的子矩阵中元素和:
差分
一维:通过空间换时间适用于需要频繁进行区间数值的批量增减操作的场景。
一维数组:
一维差分数组中的每一项:
一维差分数组的构造:
边输入原数组边构造
cpp
for(int i=1;i<=n;i++){
scanf("%d",arr+i);
diff[i]=arr[i]-arr[i-1];
}
一维差分数组与原数组的推导关系: 由差分数组定义移项可得:
一维差分数组实现区间批量增减:
让区间 [ l , r ] 上的元素加x:让 加上x,让 减去x
这种区间批量增减操作是静态的无法一遍查询一边修改,需要统一在差分数组上修改完毕后再通过差分数组与原数组的关系还原回原数组,再实现查询。
二维:等待补充。。。。