莫比乌斯二连 HDU 5212 Code & HDU 1695 GCD

莫比乌斯的模板题

都是差不多的

F(m)为gcd(i,j) = m(i∈[1,m],j∈[1,n])的个数

f(m) = ∑(m\d) F(d)  意义为gcd(i,j)为m的倍数的个数

运用莫比乌斯反演得到

F(m) = ∑(m\d)μ(d/m) * f(d)

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long LL;

#define MOD 10007
const int ListSize = 10005;
int PrimeSize,n;
int isPrime[ListSize],Mu[ListSize],Prime[ListSize];
int total[ListSize],f[ListSize],a[ListSize];

void Init(){
  Mu[1] = 1;
  memset(isPrime,1,sizeof(isPrime));
  for (int i=2;i<=ListSize;i++){
    if (isPrime[i]){
      PrimeSize ++;
      Prime[PrimeSize] = i;
      Mu[i] = -1;
    }
    for (int j=1;j<=PrimeSize && Prime[j] * i<= ListSize;j++){
      isPrime[i*Prime[j]] = 0;
      if (i % Prime[j] == 0){
          Mu[i*Prime[j]] = 0;
          break;
      }
      else {
         Mu[i*Prime[j]] = -Mu[i];
      }
    }
  }
}

int main(){
  // freopen("test.in","r",stdin);
  Init();
  while (cin >> n){
    int maxn = 0;
    memset(total,0,sizeof(total));
    memset(f,0,sizeof(f));

    for (int i=0;i<n;i++){
      scanf("%d",&a[i]);
      total[a[i]] ++;
      maxn = max(maxn,a[i]);
    }
    for (int i=1;i<=maxn;i++){
      for (int j=i;j<=maxn;j+=i){
        f[i] += total[j];
      }
    }
    LL ans = 0,temp;
    for (int i=1;i<=maxn;i++){
      temp = 0;
      for (int j=i;j<=maxn;j+=i){
        temp = (temp + Mu[j/i] * f[j] * f[j] % MOD) % MOD;
      }
      ans = (ans + temp * 1ll * i % MOD * (i-1) % MOD) % MOD;
    }
    cout << ans << endl;
  }

}

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long LL;

const int ListSize = 100005;
int PrimeSize;
int isPrime[ListSize],Mu[ListSize],Prime[ListSize];
int Sum[ListSize];
int a,b,c,d,k,T;
void Init(){
  Mu[1] = 1; Sum[0] = 0;
  memset(isPrime,1,sizeof(isPrime));
  for (int i=2;i<=ListSize;i++){
    if (isPrime[i]){
      PrimeSize ++;
      Prime[PrimeSize] = i;
      Mu[i] = -1;
    }
    for (int j=1;j<=PrimeSize && Prime[j] * i<= ListSize;j++){
      isPrime[i*Prime[j]] = 0;
      if (i % Prime[j] == 0){
          Mu[i*Prime[j]] = 0;
          break;
      }
      else {
         Mu[i*Prime[j]] = -Mu[i];
      }
    }
  }
  for (int i=1;i<ListSize;i++){
    Sum[i] = Sum[i-1] + Mu[i];
  }
}

int main(){
  // freopen("test.in","r",stdin);
  Init();
  cin >> T;
  for (int times = 1; times <= T; times ++){
    cin >> a >> b >> c >> d >> k;
    cout << "Case " << times << ": ";
    if (k == 0){
      cout << "0" << endl; continue;
    }

    b = b / k;
    d = d / k;

    if (b > d){
      swap(b,d);
    }

    LL ans1 = 0;
    int last;
    for (int i=1;i<=b;i=last+1){
      last = min(b/(b/i),d/(d/i));
      ans1 += (LL)(Sum[last] - Sum[i-1]) * (b/i) * (d/i);
    }
    LL ans2 = 0;
    for (int i=1;i<=b;i=last+1){
      last = b/(b/i);
      ans2 += (LL) (Sum[last] - Sum[i-1]) * (b/i) * (b/i);
    }
    LL ans = ans1 - ans2 / 2;
    cout << ans << endl;
  }

}

时间: 2024-10-28 11:22:22

莫比乌斯二连 HDU 5212 Code & HDU 1695 GCD的相关文章

hdu 5212 Code 筛法或者莫比乌斯

Code Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) Problem Description WLD likes playing with codes.One day he is writing a function.Howerver,his computer breaks down because the function is too powerful.He is ve

hdu.5212.Code(莫比乌斯反演 &amp;&amp; 线性筛)

Code Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) Total Submission(s): 300    Accepted Submission(s): 124 Problem Description WLD likes playing with codes.One day he is writing a function.Howerver,his computer b

HDU 5212 Code(容斥 或 莫比乌斯反演)

Code Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) Problem Description WLD likes playing with codes.One day he is writing a function.Howerver,his computer breaks down because the function is too powerful.He is ve

HDU 1695 GCD 欧拉函数+容斥原理+质因数分解

链接:http://acm.hdu.edu.cn/showproblem.php?pid=1695 题意:在[a,b]中的x,在[c,d]中的y,求x与y的最大公约数为k的组合有多少.(a=1, a <= b <= 100000, c=1, c <= d <= 100000, 0 <= k <= 100000) 思路:因为x与y的最大公约数为k,所以xx=x/k与yy=y/k一定互质.要从a/k和b/k之中选择互质的数,枚举1~b/k,当选择的yy小于等于a/k时,可以

HDU 1695 GCD (数论-整数和素数,组合数学-容斥原理)

GCD Problem Description Given 5 integers: a, b, c, d, k, you're to find x in a...b, y in c...d that GCD(x, y) = k. GCD(x, y) means the greatest common divisor of x and y. Since the number of choices may be very large, you're only required to output t

HDU 2017 Code Lock (并查集的应用+快速幂)

链接:HDU 3461 题目大意: 题目的大意是一个密码锁上有编号为1到N的N个字母,每个字母可以取26个小写英文字母中的一个.再给你M个区间[L,M],表示该区间的字母可以一起同步"增加"(从'a'变为'b'为增1,'z'增1为'a').假如一组密码按照给定的区间进行有限次的"增加"操作后可以变成另一组密码,那么我们认为这两组密码是相同的.该题的目标就是在给定N.M和M个区间的前提下计算有多少种不同的密码. 根据题意,如果一个可调整的区间都没有的话,答案应该是26

HDU 3461 Code Lock(并查集的应用+快速幂)

* 65536kb,只能开到1.76*10^7大小的数组.而题目的N取到了10^7,我开始做的时候没注意,用了按秩合并,uset+rank达到了2*10^7所以MLE,所以貌似不能用按秩合并. 其实路径压缩也可以不用.............  题目的大意: 一个密码锁上有编号为1到N的N个字母,每个字母可以取26个小写英文字母中的一个.再给你M个区间[L,M],表示该区间的字母可以一起同步"增加"(从'a'变为'b'为增1,'z'增1为'a').假如一组密码按照给定的区间进行有限

hdu 5381 The sum of gcd(线段树+gcd)

题目链接:hdu 5381 The sum of gcd 将查询离线处理,按照r排序,然后从左向右处理每个A[i],碰到查询时处理.用线段树维护,每个节点表示从[l,i]中以l为起始的区间gcd总和.所以每次修改时需要处理[1,i-1]与i的gcd值,但是因为gcd值是递减的,成log级,对于每个gcd值记录其区间即可.然后用线段树段修改,但是是修改一个等差数列. #include <cstdio> #include <cstring> #include <vector>

HDU 4910 HDOJ Problem about GCD BestCoder #3 第四题

首先 m = 1 时 ans = 0对于 m > 1 的 情况 由于 1 到 m-1 中所有和m互质的数字,在 对m的乘法取模 运算上形成了群 ai = ( 1<=a<m && gcd(a,m) == 1 ) 所以 对于 a 必然存在b = a^(-1) = inv(a) 使得 a * b = 1 (mod m) 这里存在两种情况 a != b 那么最后的连乘式中a b均出现一次,相乘得1 a == b 那么最后的连乘式中只出现一个a 实际上所有 a = inv(a) 的