数论 + 贪心 - Basic Gcd Problem - 2020牛客暑期多校训练营(第四场)

数论 + 贪心 - Basic Gcd Problem - 2020牛客暑期多校训练营(第四场)

题意:

给定表达式:

{ f c ( x ) = m a x 1 ≤ i ≤ x − 1 c ⋅ f c ( g c d ( i , x ) ) x > 1 f c ( x ) = 1 x = 1 \begin{cases}f_c(x)=max_{1≤i≤x-1}c·f_c(gcd(i,x))\qquad x>1\\\\f_c(x)=1 \qquad\qquad\qquad\qquad\qquad\qquad\quad x=1\end{cases} fc(x)=max1ix1cfc(gcd(i,x))x>1fc(x)=1x=1

T 组 测 试 数 据 , 每 组 包 括 两 个 正 整 数 n 和 c , 输 出 f c ( n ) 。 答 案 对 1 0 9 + 7 取 模 。 T组测试数据,每组包括两个正整数n和c,输出f_c(n)。答案对10^9+7取模。 Tncfc(n)109+7

数据范围:

1 ≤ T ≤ 1 0 6 , 1 ≤ n , c ≤ 1 0 6 1≤T≤10^6,1≤n,c≤10^6 1T1061n,c106

示例1
输入

2
3 3
10 5

输出

3
25

分析:

容 易 分 析 得 出 , f 迭 代 的 次 数 k 越 多 越 好 , 答 案 为 c k 。 容易分析得出,f迭代的次数k越多越好,答案为c^k。 fkck

贪 心 地 考 虑 , 分 解 n = P 1 a 1 P 2 a 2 . . . P k a k , 假 设 P 1 < P 2 < . . . < P k 贪心地考虑,分解n=P_1^{a_1}P_2^{a_2}...P_k^{a_k},假设P_1<P_2<...<P_k n=P1a1P2a2...PkakP1<P2<...<Pk

要 使 g c d 的 迭 代 次 数 最 多 , 最 优 的 情 况 就 是 每 次 约 去 一 个 最 小 质 因 子 , 这 样 总 次 数 即 所 有 质 因 子 指 数 之 和 。 要使gcd的迭代次数最多,最优的情况就是每次约去一个最小质因子,这样总次数即所有质因子指数之和。 使gcd

问题:如何快速分解正整数n,得到其所有质因子的指数之和?

本 题 T ≤ 1 0 9 , n ≤ 1 0 6 , 若 对 每 个 n 逐 个 分 解 质 因 数 , 时 间 复 杂 度 为 O ( n ) 也 还 是 会 超 时 , 本题T≤10^9,n≤10^6,若对每个n逐个分解质因数,时间复杂度为O(\sqrt{n})也还是会超时, T109n106nO(n )

因 此 只 能 实 现 预 处 理 出 来 。 因此只能实现预处理出来。

设 M [ i ] 表 示 正 整 数 i 的 所 有 质 因 子 的 指 数 之 和 。 设M[i]表示正整数i的所有质因子的指数之和。 M[i]i

① 、 若 i 是 质 数 , 则 M [ i ] = 1 。 ①、若i是质数,则M[i]=1。 iM[i]=1

② 、 对 任 意 合 数 , 分 解 其 最 小 质 因 子 P j 与 另 一 个 数 i 的 乘 积 : P j × i , 则 M [ P j × i ] + = M [ i ] + 1 。 ②、对任意合数,分解其最小质因子P_j与另一个数i的乘积:P_j×i,则M[P_j×i]+=M[i]+1。 PjiPj×iM[Pj×i]+=M[i]+1

故 我 们 可 以 线 性 筛 中 递 推 求 出 M 数 组 。 故我们可以线性筛中递推求出M数组。 线M

代码:

#include <cstdio>
#include <cstring>
#include <cmath>
#include <iostream>
#include <algorithm>
  
#define ll long long
  
using namespace std;
  
const int N=1e6+10, mod=1e9+7;

int primes[N],cnt,M[N];
bool st[N];

void get_prime(int n)
{
    for(int i=2;i<=n;i++)
    {
        if(!st[i]) 
        {
            primes[cnt++]=i;
            M[i]=1;
        }
        for(int j=0;primes[j]*i<=n;j++)
        {
            st[primes[j]*i]=true;
            M[primes[j]*i]+=M[i]+1;
            if(i%primes[j]==0) break;
        }
    }
}

int quick_pow(int a,int b)
{
    int res=1;
    while(b)
    {
        if(b&1) res=(ll)res*a%mod;
        a=(ll)a*a%mod;
        b>>=1;
    }
    return res;
}

int main()
{
    get_prime(1e6);
    int T;
    int n,c;
    cin>>T;
    while(T--)
    {
        scanf("%d%d",&n,&c);
        int k=M[n];
        printf("%d\n",quick_pow(c,k));
    }
    
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值