这里维护i表示当前结尾位置只需找到快速找出最大开头j即可,考虑不同的gcd(aj,aj+1,...ai)会从后往前保持gcd值或者是变为原来的约数,
所以维护二元组(j,g) (代表gcd为g的最小位置)则序列长度最大不超过(log(maxn) = log(10^12))非常短,所以我直接采取循环刷新;每次刷新即对g = gcd(g,a[i]); 并删除g相同的元素,取最优;
#include <cstdio> #include <vector> #include <cmath> #include <iostream> #include <cstring> using namespace std; #define INF 200000000 typedef long long LL; struct node{ long long p,g; node(long long x=0,long long y=0):p(x),g(y){} }; vector<node> pre,now; long long gcd(long long a,long long b){ return b==0 ? a : gcd(b,a%b); } const int maxn = 100100; long long a[maxn]; int main() { int n; int T; scanf("%d",&T); while(T--){ scanf("%d",&n); for(int i=1;i<=n;i++) scanf("%lld",&a[i]); pre.clear(); now.clear(); LL res=0; for(LL i=1;i<=n;i++){ now.clear(); for(LL j=0;j<pre.size();j++){ LL te = gcd(a[i],pre[j].g); if(!now.empty()&&now.back().g==te) continue; now.push_back(node(pre[j].p,te)); } if(now.empty()||a[i]>now.back().g){now.push_back(node(i,a[i]));} for(LL j=0;j<now.size();j++){ res=max(res,now[j].g*(i-now[j].p+1)); } pre.clear(); pre=now; } printf("%lld\n",res); } return 0; }