看题我想到了(n^2)预处理(O(1))回答,然后开桶,想到这里发现不会了,(n^3q)暴力相信大家都会吧
考虑弱化,(f[i][j])表示区间([l,r])中满足(kin(l,r),a[k]+a[l]+a[r]=0的k的数量),我们可以枚举左右端点开桶,(n^2预处理f)
但本题最后要求的是:在一段区间内,左右端点不强制选的方案数
考虑先求出(s[l][r])表示左端点在([1,l])内,右端点在([1,r])内方案数,这就是(f)的二维前缀和
#include<cstdio>
#define maxn 5001
#define K 1000000
#define ll long long
using namespace std;
int n,q,a[maxn],cnt[2*K+1];
ll s[maxn][maxn];
int main(){
scanf("%d%d",&n,&q);
for(int i = 1;i <= n;i++)
{scanf("%d",&a[i]);a[i] += K;}
for(int i = 1;i <= n;i++){
for(int j = i+1;j <= n;j++){
if(j > i+1){
if(a[i] + a[j] <= K*3 && a[i]+a[j] >= K)
s[i][j] = cnt[K*3 - a[i] - a[j]];}
cnt[a[j]] ++;
}
for(int j = i+1;j <= n;j++)
cnt[a[j]] --;
}
for(int i = 1; i <= n; ++i)
for(int j = 1; j <= n; ++j)
s[i][j] += s[i-1][j]+s[i][j-1]-s[i-1][j-1];
int l,r;
while(q--){
scanf("%d%d",&l,&r);
printf("%lld
", s[r][r]-s[l-1][r]-s[r][l-1]+s[l-1][l-1]);
}
}