zoukankan      html  css  js  c++  java
  • luogu4917天守阁的地板

    https://www.zybuluo.com/ysner/note/1317548---

    题面

    给出(n),用所有长为(a)、宽为(b)((1leq a,bleq n))的长方形拼成正方形,最少需多少块?
    多组数据。

    • (30pts) (nleq100,Tleq100)
    • (60pts) (nleq3*10^4,Tleq300)
    • (100pts) (a,bleq10^5,Tleq1000)

    解析

    显然答案是$$frac{lcm(a,b)}{a}*frac{lcm(a,b)}{b}$$
    暴力复杂度(O(Tn^2logn)),可以通过(30pts)

    考虑推推柿子。

    [prod_{a=1}^nprod_{b=1}^nfrac{lcm(a,b)}{a}*frac{lcm(a,b)}{b} ]

    [=prod_{a=1}^nprod_{b=1}^nfrac{lcm^2(a,b)}{ab} ]

    [=prod_{a=1}^nprod_{b=1}^nfrac{ab}{gcd^2(a,b)} ]

    [=frac{(n!)^{2n}}{prod_{a=1}^nprod_{b=1}^ngcd^2(a,b)} ]

    现在问题是(prod_{a=1}^nprod_{b=1}^ngcd(a,b))
    这个复杂度(O(n^2)),很不划算。
    考虑枚举最大公约数的值。
    则化为

    [prod_{d=1}^nd^{sum_{a=1}^nsum_{b=1}^n[gcd(a,b)==d]} ]

    [prod_{d=1}^nd^{sum_{a=1}^{lfloorfrac{n}{d} floor}sum_{b=1}^{lfloorfrac{n}{d} floor}[gcd(a,b)==1]} ]

    这个指数怎么化呢?
    我们可以强制(a>b),那么指数为(sum_{a=1}^{lfloorfrac{n}{d} floor}varphi(a))
    如果不强制,考虑到(varphi(1)=1)的特殊情况,柿子可化为:

    [prod_{d=1}^nd^{[sum_{a=1}^{lfloorfrac{n}{d} floor}2*varphi(a)]-1} ]

    然后线性预处理一下欧拉函数前缀和,这样复杂度(O(Tn))(60pts)稳了。

    然后吗,注意到(lfloorfrac{n}{d} floor)在一段区间内是相同的,可以数论分块。
    于是就做完了。
    预处理逆元后,复杂度(O(Tsqrt n))。

    #include<iostream>
    #include<cmath>
    #include<cstdio>
    #include<cstdlib>
    #include<cstring>
    #include<algorithm>
    #define ll long long
    #define re register
    #define il inline
    #define fp(i,a,b) for(re int i=a;i<=b;i++)
    #define fq(i,a,b) for(re int i=a;i>=b;i--)
    using namespace std;
    const int mod=19260817,N=1e6+100;
    int a,b,jc[N],pri[N],ol[N],n,tot,inv[mod+100];
    ll ans,gu;
    bool vis[N];
    il ll gi()
    {
      re ll x=0,t=1;
      re char ch=getchar();
      while(ch!='-'&&(ch<'0'||ch>'9')) ch=getchar();
      if(ch=='-') t=-1,ch=getchar();
      while(ch>='0'&&ch<='9') x=x*10+ch-48,ch=getchar();
      return x*t;
    }
    il ll ksm(re ll S,re ll n)
    {
      re ll T=S;S=1;
      if(n<0) return 0;
      while(n)
        {
          if(n&1) S=S*T%mod;
          T=T*T%mod;
          n>>=1;
        }
      return S;
    }
    il void Pre(re int n)
    {
      ol[1]=1;
      fp(i,2,n)
        {
          if(!vis[i]) pri[++tot]=i,ol[i]=i-1;
          for(re int j=1;j<=tot&&i*pri[j]<=n;++j)
        {
          vis[i*pri[j]]=1;
          if(i%pri[j]) ol[i*pri[j]]=ol[i]*(pri[j]-1);
          else {ol[i*pri[j]]=ol[i]*pri[j];break;}
        }
        }
      fp(i,1,n) (ol[i]+=ol[i-1])%=(mod-1);
      jc[0]=1;fp(i,1,n) jc[i]=1ll*jc[i-1]*i%mod;
      inv[0]=inv[1]=1;fp(i,2,mod) inv[i]=1ll*(mod-mod/i)*inv[mod%i]%mod;
    }	  
    int main()
    {
      re int T=gi();
      Pre(1e6);
      while(T--)
        {
          n=gi();
          ans=ksm(jc[n],2*n);gu=1;
          re int L=0;
          for(re int i=1;i<=n;i=L+1)
        {
          L=n/(n/i);
          (gu*=ksm(1ll*jc[L]*inv[jc[i-1]]%mod,2*ol[n/i]-1))%=mod;
        }
          (ans*=inv[gu*gu%mod])%=mod;
          printf("%lld
    ",ans);
        }
      return 0;
    }
    
  • 相关阅读:
    [Swift]LeetCode32. 最长有效括号 | Longest Valid Parentheses
    [Swift]LeetCode31. 下一个排列 | Next Permutation
    [Swift]LeetCode30. 与所有单词相关联的字串 | Substring with Concatenation of All Words
    [Swift]LeetCode29. 两数相除 | Divide Two Integers
    时光轴的设计理念
    ITFriend开发日志20140611
    ITFriend开发日志20140611
    高中生活--第7篇–我为什么不交作业
    高中生活--第7篇–我为什么不交作业
    ITFriend网站内测公测感悟
  • 原文地址:https://www.cnblogs.com/yanshannan/p/9831937.html
Copyright © 2011-2022 走看看