Never gonna say goodbye
RedLycoris · · 题解
link
题目大意:
给出一个长度为
题解:
考虑我们固定一个起点
显然,这个 gcd 最多只会变化
所以我们可以考虑枚举开头,每次二分出下一个变化点的位置,每个开头二分
注意用
Code:
#include<bits/stdc++.h>
#define ll long long
using namespace std;
const int mxn=1e6+6;
int st[mxn][20],n,q,a[mxn],lg[mxn];
inline int ask(int l,int r){
int t=lg[r-l+1];
return __gcd(st[l][t],st[r-(1<<t)+1][t]);
}
map<int,ll>cnt;
int main(){
ios_base::sync_with_stdio(false);
cin.tie(0),cout.tie(0);
cin>>n>>q;
for(int i=1;i<=n;++i)cin>>a[i],st[i][0]=a[i];
for(int i=2;i<=n;++i)lg[i]=lg[i>>1]+1;
for(int j=1;j<20;++j)for(int i=1;i<=n;++i)st[i][j]=__gcd(st[i][j-1],st[i+(1<<(j-1))][j-1]);
for(int i=1;i<=n;++i){
int cur=i,val=a[i];
while(1){
int lo=cur,hi=n+1,md;
for(;lo<hi-1;){
md=lo+hi>>1;
if(ask(i,md)!=val)hi=md;
else lo=md;
}
cnt[val]+=hi-cur;
cur=hi;val=ask(i,hi);
if(hi==n+1)break;
}
}
for(;q--;){
int x;cin>>x;
cout<<cnt[x]<<'\n';
}
}