标签:各路 long map main 判断 out 子序列 stdout highlight
什么沙雕题啊……考察的是啥啊,分类咋搞啊……愁死我了……
先把作者的正解放出来:
序列
因为选出的一段是一个等比序列的子序列,我们分为两种情况:
1. q=1,相当于找一个最长每个数都相等的子串,这个扫一遍就行了。
2. q!=1,那么这个序列最长只有 logn,那么我们可以枚举开头,不妨设开始的两个数为 a[i]和 a[i+1],
其中比较大的一个为 x,另一个 y。
(1) 首先要满足 x%y=0
(2) 让 z=x/y,然后把 z 质因数分解,z=p1^q1*p2^q2*p3^q3......,设 g=gcd(q1,q2,q3...),那么当前序
列的最小公比就是 p1^(q1/g)*p2^(q2/h)*......
(3) 我们找到最小公比后,每当往后加一个数,判断它与前边的数的比是否是最小公比的整次幂,
不是的话就说明不能再加了。
(4) 还有一个要求就是这个序列里不能有重复的数,这个东西用 set 判断就行了。
复杂度 nlog2 n
Ps:因为这道题的数据实在是不好造,所以最后公比最大不会很大,可能有很多其他算法也能跑过去。
然而各路大佬各种nb的暴力卡数据都A掉了这题,(只有我这个蒟弱暴力都能爆0……)
首先求出q=1时的答案ans,
枚举q1~1000,如果ans个q乘起来超过了数列的最大值,break;
然后就是一个&n^2&的暴力,用map处理重复的数,复杂度好高……
#include<algorithm> #include<iostream> #include<cstring> #include<cstdio> #include<cmath> #include<set> #include<map> #define int LL #define LL long long #define ma(x) memset(x,0,sizeof(x)) #define max(a,b) ((a)>(b)?(a):(b)) using namespace std; int n; LL A[200010]; LL l=1,r=100000,mid,q; LL ans=1,maxn; map<int,int> mp; inline LL read() { LL s=0;char a=getchar(); while(a<‘0‘||a>‘9‘)a=getchar(); while(a>=‘0‘&&a<=‘9‘){s=s*10+a-‘0‘;a=getchar();} return s; } LL gcd(int a,int b){return !b?a:gcd(b,a%b);} signed main() { // freopen("in.txt","r",stdin); // freopen("0.out","w",stdout); n=read(); for(int i=1;i<=n;i++)A[i]=read(),maxn=max(maxn,A[i]); int L=0,R=0;r=n; for(int i=1;i<=n;i++) { if(A[i]==A[i-1]&&!L)L=i-1; if(A[i]!=A[i-1]&&L)ans=max(ans,i-L),L=i; } if(A[n]==A[n-1])ans=max(ans,n-L+1); bool pd=0; for(int q=2;q<=100;q++) { int tem=0; for(int i=1;i<=ans;i++) { tem*=q;if(q>maxn){pd=1;break;} } if(pd)break; for(int i=1;i<=n;i++) { mp.clear(); int tem=A[i]; mp[tem]=1; if(tem%q!=0&&tem>q){continue;} while(tem%q==0)tem/=q; // while(tem>=q)tem/=q; for(int j=i+1;j<=n;j++) { int te=A[j]; if(mp.find(te)!=mp.end()){ans=max(ans,j-i);break;} if(te%q!=0&&te!=tem){ans=max(ans,j-i);break;} while(te%q==0)te/=q; if(te!=tem){ans=max(ans,j-i);break;} } } } printf("%lld\n",ans); }
标签:各路 long map main 判断 out 子序列 stdout highlight
原文地址:https://www.cnblogs.com/Al-Ca/p/11186759.html