素数

素数

一些定理性质

  • 唯一分解定理

  • 威尔逊定理:

    \(p\)是质数,则有\((p - 1) ! \equiv -1 \pmod p\)

    逆定理同时成立,如果有\((p - 1) ! \equiv -1 \pmod p\),则\(p\)一定是质数。

    \(((p - 1) ! + 1) \equiv 0 \pmod p\)可以通过\(\sin(\pi((n - 1)! + 1) / n) = 0\)来得到素数所在点函数。

  • 费马小定理:

    假设\(p\)是一个质数,\(a\)不是\(p\)的倍数,则有\(a^{p - 1} \equiv 1\pmod p\),证明如下

    我们分别取\(i * a \pmod p\)的值,\(for i in range(1, p)\),由于\(i\)\(p\)互质,\(a\)\(p\)互质,所以,这些取模后的值是一个\((1 ~ n - 1)\)的排列,那么另这\(p - 1\)个数相乘,得到的就是\(a ^ {p - 1} * (p - 1)! \pmod p = (p - 1)!\),因为\((p - 1) ! \pmod p != 0\),所以两边同时约去得到\(a ^ {p - 1} \equiv 1 \pmod p\)

Miller_Rabin素数测试

利用了\(a^{p} \equiv a \pmod p\),这一性质来对素数进行判定,但是具体的证明我也没仔细看,就是测试一个数字之后就能有\(1/ 4\)的概率证明其不是质数,然后通过多测试几个数,就可以几乎精确的得到这个数是不是质数。

#include 
#define mp make_pair
#define pb push_back

using namespace std;

typedef pair pii;
typedef long long ll;
typedef unsigned long long ull;

const double eps = 1e-7;
const double pi = acos(-1.0);
const int inf = 0x3f3f3f3f;

inline ll read() {
    ll f = 1, x = 0;
    char c = getchar();
    while(c < '0' || c > '9') {
        if(c == '-') f = -1;
        c = getchar();
    } 
    while(c >= '0' && c <= '9') {
        x = (x << 1) + (x << 3) + (c ^ 48);
        c = getchar();
    }
    return f * x;
}


ll qpow(ll a, ll n, ll mod) {
    ll ans = 1;
    while(n) {
        if(n & 1)   ans = (ans * a) % mod;
        n >>= 1;
        a = (a * a) % mod;
    }
    return ans;
}

bool miller_rabin(ll a) {
    if(a == 2)  return true;
    for(int i = 1; i <= 10; i++) {
        ll temp = rand() % (a - 2) + 2;
        if(qpow(temp, a, a) != temp)    return false;
    }
    return true;
}

int main() {
    // freopen("in.txt", "r", stdin);
    // freopen("out.txt", "w", stdout);
    // ios::sync_with_stdio(false), cin.tie(0), cout.tie(0);
    srand(time(0));
    ll a;
    while(scanf("%lld", &a) != EOF) {
        if(miller_rabin(a)) puts("prime");
        else    puts("not");
    }
    return 0;
}

欧拉公式与欧拉定理

欧拉公式\(\phi(n) = x\)\(x\)是小于\(n\)并且与\(n\)互质的数字的个数,有\(\phi(8) = 4 (1, 3, 5, 7)\)

欧拉公式成立的几个定理:

  • 如果\(p\)是质数,则有\(\phi(p) = p - 1\),这是显然成立的。

  • 如果\(p\)是质数,则\(\phi(p^i) = (p - 1) * p ^ {i - 1}\),证明:

    \([1, p ^ i)\)之间,与\(p\)互质的数有\(p* t(t = 1, 2, 3, …… p ^ {i - 1} - 1)\),所以\(\phi(p^i) = p ^ i - 1 - (p ^ {i - 1} - 1) = (p - 1) * p ^ {i - 1}\)

  • \(\phi(n) = n * (1 - 1 / p_1) * (1 - 1 / p_2) …… (1 - 1 / p_{n - 1}) * (1 - 1 / p_{n})\)\(p_i\)\(n\)的质因数,例如:\(8 = 2 ^ 3, \phi(8) = 8 * (1 - 1 / 2) = 4\)

  • 如果\(a, b\)互质,则有\(\phi(a * b) = \phi(a) * \phi(b)\),证明:

    可用上述的第三个定理来说明,\(a, b\)中的质因数是互质的,所以,任何两个\(p_{i}\)是互质的,然后显然成立。

欧拉定理

\(a, m\)互质有\(a ^ {\phi(m)} \equiv 1 \pmod m\),当\(m\)时质数的时候,也就是\(a ^ {m-1} \equiv 1 \pmod m\)

欧拉筛(素数+欧拉函数)

最后给上素数筛和欧拉函数的线性求解法。

#include 
#define mp make_pair
#define pb push_back

using namespace std;

typedef pair pii;
typedef long long ll;
typedef unsigned long long ull;

const double eps = 1e-7;
const double pi = acos(-1.0);
const int inf = 0x3f3f3f3f;

inline ll read() {
    ll f = 1, x = 0;
    char c = getchar();
    while(c < '0' || c > '9') {
        if(c == '-') f = -1;
        c = getchar();
    } 
    while(c >= '0' && c <= '9') {
        x = (x << 1) + (x << 3) + (c ^ 48);
        c = getchar();
    }
    return f * x;
}

const int N = 1e5 + 10;

int prime[N], phi[N], cnt = 0;
bool st[N];

void init() {
    phi[1] = 1;
    for(int i = 2; i < N; i++) {
        if(!st[i]) {
            // cout << i << endl;
            prime[cnt++] = i;
            phi[i] = i - 1;
        }
        for(int j = 0; j < cnt && i * prime[j] < N; j++) {
            st[i * prime[j]] = 1;
            if(i % prime[j] == 0) {
                phi[i * prime[j]] = phi[i] * prime[j];
                break;
            }
            phi[i * prime[j]] = phi[i] * (prime[j] - 1);
        }
    }
}


int main() {
    // freopen("in.txt", "r", stdin);
    // freopen("out.txt", "w", stdout);
    // ios::sync_with_stdio(false), cin.tie(0), cout.tie(0);
    init();
    return 0;
}

你可能感兴趣的:(素数)