题面:https://www.cnblogs.com/Juve/articles/11648975.html
神炎皇:
打表找规律?和$phi$有关?
答案就是$sumlimits_{i=2}^{n}phi(i)*frac{n}{i*i}$

1 #include<iostream> 2 #include<cstdio> 3 #include<cstring> 4 #include<algorithm> 5 #include<cmath> 6 #define int long long 7 using namespace std; 8 int n,ans=0; 9 int prime[2000006],tot=0,phi[10000006]; 10 bool vis[10000007]; 11 void get_phi(int N){ 12 vis[1]=phi[1]=1; 13 for(int i=2;i<=N;i++){ 14 if(!vis[i]) prime[++tot]=i,phi[i]=i-1; 15 for(int j=1;j<=tot&&i*prime[j]<=N;j++){ 16 vis[i*prime[j]]=1; 17 if(!(i%prime[j])){ 18 phi[i*prime[j]]=phi[i]*prime[j]; 19 break; 20 } 21 phi[i*prime[j]]=phi[i]*phi[prime[j]]; 22 } 23 } 24 } 25 signed main(){ 26 scanf("%lld",&n); 27 get_phi(sqrt(n)+1); 28 for(int i=2;i<=sqrt(n);++i) 29 ans+=phi[i]*(n/(i*i)); 30 printf("%lld ",ans); 31 return 0; 32 }
降雷皇
基础不牢,地动山摇。。。赶紧补一发LIS
给定序列$a_i$,设f[i]表示以a[i]结尾的最长上升序列长度
则$f[i]=max(f[j]+1)(j<i&&a[j]<a[i])$
设g[i]表示以i为结尾的最长上升序列的个数
则有$g[i]=sumlimits_{j=1}^{i-1}[a[j]<a[i]&&f[j]=f[i]-1]*g[j]$
这样就有了$O(n^2)$转移
然后用树状数组加速转移
树状数组下标为权值,顺序扫保证了j<i,树状数组查询a[i]-1保证了a[j]<a[i],然后树状数组即可
当然也可以线段树,原理是一样的
#include<iostream> #include<cstdio> #include<cstring> #include<algorithm> using namespace std; const int MAXN=1e5+5; const int mod=123456789; int n,typ,a[MAXN],mx=0; pair<int,int>c[MAXN*10],f[MAXN],ans;//changdu,geshu int lowbit(int x){ return x&(-x); } int update(int pos,pair<int,int> val){ for(int i=pos;i<=mx;i+=lowbit(i)){ if(c[i].first<val.first) c[i]=val; else if(c[i].first==val.first) (c[i].second+=val.second)%=mod; } } pair<int,int> query(int pos){ pair<int,int>res; res.first=0,res.second=1; for(int i=pos;i>0;i-=lowbit(i)){ if(res.first<c[i].first) res=c[i]; else if(res.first==c[i].first) (res.second+=c[i].second)%=mod; } return res; } int main(){ scanf("%d%d",&n,&typ); for(int i=1;i<=n;++i) scanf("%d",&a[i]),mx=max(mx,a[i]); for(int i=1;i<=n;++i){ f[i]=query(a[i]-1); ++f[i].first; update(a[i],f[i]); if(ans.first<f[i].first) ans=f[i]; else if(ans.first==f[i].first) (ans.second+=f[i].second)%=mod; } if(typ==0) printf("%d ",ans.first); else printf("%d %d ",ans.first,ans.second); return 0; }
幻魔皇:
产生贡献的只有白点,我们按两个白点lca的颜色分类
如果两个白点lca是白点,那么这个白点一定是两个白点中的一个
枚举两个白点的距离,不难发现只有深度在[1,n-i]的白点在它的子树中有距离为i的白点
设sum[i]表示前i层白点个数,w[i]表示第i层白点个数,因为每个白点的子树结构都相同,所以每个白点所在子树中第i层的白点个数都相等
所以贡献就是sum[n-i]*w[i],
如果两个白点的lca是黑点,枚举两个白点到黑点的距离i,j,只有深度为[1,n-max(i,j)]的黑点有贡献
设f[i]表示前i层的黑点个数,那么答案就是f[n-max(i,j)]*w[i]*w[j+1],其中i,j是有顺序的,i表示在黑点的白儿子的子树中的点,j表示在黑点的黑儿子的子树中的点
sum[],w[],f[]用fib递推

1 #include<iostream> 2 #include<cstdio> 3 #include<cstring> 4 #include<algorithm> 5 #define int long long 6 using namespace std; 7 const int MAXN=5005; 8 const int mod=123456789; 9 int n,w[MAXN],sum[MAXN],f[MAXN],ans[MAXN<<1]; 10 signed main(){ 11 scanf("%lld",&n); 12 w[1]=w[3]=w[4]=1;w[2]=0; 13 for(int i=5;i<=n;++i) w[i]=(w[i-1]+w[i-2])%mod; 14 for(int i=1;i<=n;++i) sum[i]=(sum[i-1]+w[i])%mod; 15 f[1]=0,f[2]=f[3]=1; 16 for(int i=4;i<=n;++i) f[i]=(f[i-1]+f[i-2])%mod; 17 for(int i=1;i<=n;++i) f[i]=(f[i-1]+f[i])%mod; 18 for(int i=1;i<n;++i){ 19 (ans[i]+=(sum[n-i]*w[i+1])%mod)%=mod; 20 for(int j=1;j<n;++j) 21 (ans[i+j]+=f[n-max(i,j)]*w[i]%mod*w[j+1]%mod)%=mod; 22 } 23 for(int i=1;i<=2*n;++i) printf("%lld ",ans[i]); 24 puts(""); 25 return 0; 26 }