http://codeforces.com/problemset/problem/889/E
题解
首先我们观察到在每次取模的过程中一定会有一次的结果是\(a_i-1\),因为如果不是,我们可以调整,答案肯定是会更优的。
于是我们的有用状态就变成了\(O(n)\)级别。
我们可以对于一个状态,把它表示为\((a,b)\),表示前\(i\)个数,当前取完模的结果为\(a\),总和写成\(i*a+b\)的形式后最大的\(b\)。
我们的转移每次要把\(a\)变成\(a%v\),再添加一个新状态\(v-1\)。
转移讨论一下。
代码
#include<bits/stdc++.h> #define N 200009 using namespace std; typedef long long ll; const int mod=1e9+7; map<ll,ll>f; map<ll,ll>::iterator it; int n; inline ll rd(){ ll x=0;char c=getchar();bool f=0; while(!isdigit(c)){if(c=='-')f=1;c=getchar();} while(isdigit(c)){x=(x<<1)+(x<<3)+(c^48);c=getchar();} return f?-x:x; } int main(){ n=rd(); f[rd()-1]=0; for(int i=2;i<=n;++i){ ll x=rd(); while(!f.empty()){ it=f.end();--it; ll a=it->first,b=it->second; if(a<x)break; f.erase(it); f[x-1]=max(f[x-1],b+1ll*(i-1)*(a-a%x-x)); f[a%x]=max(f[a%x],b+1ll*(i-1)*(a-a%x)); } } ll ans=0; for(it=f.begin();it!=f.end();++it)ans=max(ans,it->first*n+it->second); cout<<ans; return 0; }
转载于:https://www.cnblogs.com/ZH-comld/p/11020728.html
发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/100867.html原文链接:https://javaforall.cn
【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛
【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...