思路:
维护一下前缀和对\(M\)取模得到的值.
考虑当一段区间的结尾固定时,如何寻找区间的起始端点才能使得答案最优.
首先是减去一个比当前小的数,为了使答案最大,当然是减去0,也就是用当前的模来更新答案;
其次是减去一个比当前大的数,这个需要加上\(M\),由于贪心的使答案最大,我们就减去一个之前出现过的比当前大的最小的模来更新答案.这东西用一个set维护就行.
然后我被迷の卡常数了QoQ.
这样写:upper_bound(s.begin(),s.end(),sum[i])就TLE.(T的不是一点半点)
这样写:s.upper_bound(sum[i])就AC.
这TM是什么鬼?
#include<cstdio> #include<cstring> #include<cctype> #include<climits> #include<set> #include<iostream> #include<algorithm> using namespace std; typedef long long LL; namespace Fio{ inline int getc(){ static const int L=1<<15;static char buf[L],*S=buf,*T=buf; if(S==T){T=(S=buf)+fread(buf,1,L,stdin);if(S==T)return EOF;} return*S++; } template<typename T>inline void Getunsign(T&x){ int c;while(!isdigit(c=getc()));x=c-'0';while(isdigit(c=getc()))x=(x<<1)+(x<<3)+c-'0'; } template<typename T>inline void Get(T&x){ int c;while(!isdigit(c=getc())&&c!='-');bool sign=c=='-'; x=sign?0:c-'0';while(isdigit(c=getc()))x=(x<<1)+(x<<3)+c-'0'; if(sign)x=-x; } } #define N 200010 LL a[N],sum[N],p; set<LL>s; int main(){ //freopen("tt.in","r",stdin); using namespace Fio; int n;Getunsign(n);Getunsign(p); register int i,j;for(i=1;i<=n;++i){Get(a[i]),a[i]%=p;if(a[i]<0)a[i]+=p;} for(i=1;i<=n;++i){sum[i]=sum[i-1],sum[i]+=a[i];if(sum[i]>=p)sum[i]-=p;} LL nowans=-1; for(i=1;i<=n;++i){ if(sum[i]>nowans)nowans=sum[i]; set<LL>::iterator it=s.upper_bound(sum[i]); if(it!=s.end()&&nowans<sum[i]-*it+p)nowans=sum[i]-*it+p; s.insert(sum[i]); } printf("%lld",nowans); return 0; }