AcWing 197. 阶乘分解

AcWing 197. 阶乘分解

一、题目描述

给定整数 N,试把阶乘 N! 分解质因数,按照算术基本定理的形式输出分解结果中的 pici 即可。

输入格式
一个整数 N

输出格式
N! 分解质因数后的结果,共若干行,每行一对 pi,ci,表示含有 pici 项。按照 pi 从小到大的顺序输出。

数据范围
3N106

输入样例
5

输出样例

2 3
3 1
5 1

样例解释

5!=120=2^335

二、方法1

#include <bits/stdc++.h>
using namespace std;
typedef long long LL;

int n;
bool prime(int x) {
    if (x == 2) return 1;
    for (int i = 2; i * i <= x; i++)
        if (x % i == 0) return 0;
    return 1;
}
int main() {
    cin >> n;
    for (int i = 2; i <= n; i++) {
        if (!prime(i)) continue;
        LL x = i;
        int ans = 0;
        while (x <= n) ans += n / x, x *= i;
        printf("%d %d\n", i, ans);
    }
    return 0;
}

三、方法2

  1. 筛出1n的所有质数
  2. 枚举每个质因子xn!表示123...n,从1n中,求x的次数:

cnt(x)=nx1+nx2+nx3+...

(直到x的次方大于n停止)
举栗子:
比如10!=1×2×3×4×5×6×7×8×9×10里有多少个2的倍数,

nx1=102=5

有多少个4的倍数

nx2=104=2

有多少个8的倍数

nx2=108=1

5+2+1=8个,与例子对的上。

Code

#include <bits/stdc++.h>
using namespace std;
typedef long long LL;

// 欧拉筛
const int N = 1e6 + 10;
int primes[N], cnt; // primes[]存储所有素数
bool st[N];         // st[x]存储x是否被筛掉
void get_primes(int n) {
    memset(st, 0, sizeof st);
    cnt = 0;
    for (int i = 2; i <= n; i++) {
        if (!st[i]) primes[cnt++] = i;
        for (int j = 0; primes[j] * i <= n; j++) {
            st[primes[j] * i] = true;
            if (i % primes[j] == 0) break;
        }
    }
}
int main() {
    int n;
    cin >> n;

    get_primes(n);

    for (int i = 0; i < cnt; i++) {
        int p = primes[i];
        int s = 0;
        // 思路:由大到小+除法降维
        // 优点:不用考虑乘法而导致的爆int上限
        for (int j = n; j; j /= p) s += j / p;
        printf("%d %d\n", p, s);
    }
    return 0;
}
posted @   糖豆爸爸  阅读(119)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 【自荐】一款简洁、开源的在线白板工具 Drawnix
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· Docker 太简单,K8s 太复杂?w7panel 让容器管理更轻松!
Live2D
点击右上角即可分享
微信分享提示