【数学】线性筛

\(i\) 的最小质因数 \(pm[i]\)
\(i\) 的最大的最小质因数的幂 \(pk[i]\)

判断 \(i\) 是不是质数的充要条件为 \(i>1 \; and \; pm[i]=i\)

const int MAXN = 1e6 + 10;
int p[MAXN], ptop;
int pm[MAXN], pk[MAXN];

void sieve(int n) {
    memset(pm, 0, sizeof(pm[0]) * (n + 1));
    ptop = 0, pm[1] = 1, pk[1] = 1;
    for(int i = 2; i <= n; ++i) {
        if(!pm[i])
            p[++ptop] = i, pm[i] = i, pk[i] = i;
        for(int j = 1, t; j <= ptop && (t = i * p[j]) <= n; ++j) {
            pm[t] = p[j];
            if(i % p[j])
                pk[t] = pk[p[j]];
            else {
                pk[t] = pk[i] * p[j];
                break;
            }
        }
    }
}

注意每个数拥有的不同的质因数也是可以线性筛的,但是这个并不是积性函数。下面用pdd来表示。

const int MAXN = 1e6 + 10;
int p[MAXN], ptop;
int pm[MAXN], pk[MAXN];
int pdd[MAXN];

void sieve (int n) {
    memset (pm, 0, sizeof (pm[0]) * (n + 1));
    ptop = 0, pm[1] = 1, pk[1] = 1;
    pdd[1] = 0;
    for (int i = 2; i <= n; ++i) {
        if (!pm[i]) {
            p[++ptop] = i, pm[i] = i, pk[i] = i;
            pdd[i] = 1;
        }
        for (int j = 1, t; j <= ptop && (t = i * p[j]) <= n; ++j) {
            pm[t] = p[j];
            if (i % p[j]) {
                pk[t] = pk[p[j]];
                pdd[t] = pdd[i] + 1;
            } else {
                pk[t] = pk[i] * p[j];
                if (pk[t] == t) {
                    pdd[t] = 1;
                } else {
                    pdd[t] = pdd[t / pk[t]];
                }
                break;
            }
        }
    }
}

数论函数

除数函数 \(\sigma_k(n)\)\(n\) 的因子的 \(k\) 次方和,当 \(k=0\) 时,为 \(n\) 的因数个数 \(d(n)\)

\(i\) 的因数个数 \(pd[i]\)
\(i\) 的最大的最小质因数的幂 \(pk[i]\)

const int MAXN = 1e6 + 10;
int p[MAXN], ptop;
int pm[MAXN], pk[MAXN], pd[MAXN];

void sieve(int n) {
    memset(pm, 0, sizeof(pm[0]) * (n + 1));
    ptop = 0, pm[1] = 1, pk[1] = 1, pd[1] = 1;
    for(int i = 2; i <= n; ++i) {
        if(!pm[i]) {
            p[++ptop] = i, pm[i] = i;
            pk[i] = i, pd[i] = 2;
        }
        for(int j = 1, t; j <= ptop && (t = i * p[j]) <= n; ++j) {
            pm[t] = p[j];
            if(i % p[j]) {
                pk[t] = pk[p[j]];
                pd[t] = pd[i] * pd[p[j]];
            } else {
                pk[t] = pk[i] * p[j];
                pd[t] = (pk[t] == t) ? pd[t / p[j]] + 1 : pd[t / pk[t]] * pd[pk[t]];
                break;
            }
        }
    }
}

Euler函数:

const int MAXN = 1e6 + 10;
int p[MAXN], ptop;
int pm[MAXN], pk[MAXN], phi[MAXN];

void sieve(int n) {
    memset(pm, 0, sizeof(pm[0]) * (n + 1));
    ptop = 0, pm[1] = 1, pk[1] = 1, phi[1] = 1;
    for(int i = 2; i <= n; ++i) {
        if(!pm[i]) {
            p[++ptop] = i, pm[i] = i;
            pk[i] = i, phi[i] = i - 1;
        }
        for(int j = 1, t; j <= ptop && (t = i * p[j]) <= n; ++j) {
            pm[t] = p[j];
            if(i % p[j]) {
                pk[t] = pk[p[j]];
                phi[t] = phi[i] * phi[p[j]];
            } else {
                pk[t] = pk[i] * p[j];
                phi[t] = (pk[t] == t) ? t - t / p[j] : phi[t / pk[t]] * phi[pk[t]];
                break;
            }
        }
    }
}

Mobius函数

const int MAXN = 1e6 + 10;
int p[MAXN], ptop;
int pm[MAXN], pk[MAXN], mu[MAXN];

void sieve(int n) {
    memset(pm, 0, sizeof(pm[0]) * (n + 1));
    ptop = 0, pm[1] = 1, pk[1] = 1, mu[1] = 1;
    for(int i = 2; i <= n; ++i) {
        if(!pm[i]) {
            p[++ptop] = i, pm[i] = i;
            pk[i] = i, mu[i] = -1;
        }
        for(int j = 1, t; j <= ptop && (t = i * p[j]) <= n; ++j) {
            pm[t] = p[j];
            if(i % p[j]) {
                pk[t] = pk[p[j]];
                mu[t] = mu[i] * mu[p[j]];
            } else {
                pk[t] = pk[i] * p[j];
                mu[t] = (pk[t] == t) ? 0 : mu[t / pk[t]] * mu[pk[t]];
                break;
            }
        }
    }
}

任意的积性函数

分为1,质数,质数的幂三种不同情形,其他的用积性函数的性质得到。

const int MAXN = 1e6 + 10;
int p[MAXN], ptop;
int pm[MAXN], pk[MAXN], f[MAXN];

void sieve(int n) {
    memset(pm, 0, sizeof(pm[0]) * (n + 1));
    ptop = 0, pm[1] = 1, pk[1] = 1, f[1] = getF1();
    for(int i = 2; i <= n; ++i) {
        if(!pm[i]) {
            p[++ptop] = i, pm[i] = i;
            pk[i] = i, f[i] = getFp(i);
        }
        for(int j = 1, t; j <= ptop && (t = i * p[j]) <= n; ++j) {
            pm[t] = p[j];
            if(i % p[j]) {
                pk[t] = pk[p[j]];
                f[t] = f[i] * f[p[j]];
            } else {
                pk[t] = pk[i] * p[j];
                f[t] = (pk[t] == t) ? getFpk(t, p[j]) : f[t / pk[t]] * f[pk[t]];
                break;
            }
        }
    }
}

GStnt柯南哥发现了一种利用最大质因数pM[i]预处理的办法,相比于最小质因数的方法来说好像节省了额外空间的使用。虽然不清楚还有什么优势(或许没有?基本上用最小质因数都可以做)不过确实挺巧妙的(复杂度不一定对?):

// 复杂度不一定对
const int MAXN = 1e6 + 10;
int p[MAXN], ptop;
int pm[MAXN], pM[MAXN];

void sieve (int n) {
    memset (pm, 0, sizeof (pm[0]) * (n + 1));
    ptop = 0, pm[1] = 1, pM[1] = 1;
    for (int i = 2; i <= n; ++i) {
        if (!pm[i]) p[++ptop] = i, pm[i] = i, pM[i] = i;
        for (int j = 1; j <= ptop && p[j] * i <= n; ++j) {
            pm[p[j] * i] = p[j];
            if (i % p[j] == 0) break;
        }
    }
    for (int i = 2, bj = 0; i <= n; ++i) {
        for (; bj + 1 <= ptop && p[bj + 1] * i <= n; ++bj);
        for (int j = bj; j >= 1 && p[j] >= pM[i]; --j) pM[p[j] * i] = p[j];
    }
}
posted @ 2020-11-10 17:39  purinliang  阅读(112)  评论(0编辑  收藏  举报