Jerry的食粮

单点时限: 2.0 sec
内存限制: 64 MB

公元9102年,Tom不再追逐Jerry,因为Tom的主人已经从China进口了很多猫粮,并且还有了智能抓老鼠机器。
小Jerry无处可躲,只能露宿街头。Tom不再缺乏食粮,但是可怜的小Jerry仍然还饥肠辘辘。
这一天,小Jerry实在是饿坏了,再不吃点儿东西,他可能就横死街头了,于是,他想到了他的老朋友Tom,
Tom这个糟老头子可是坏得很呀,他给小Jerry出了一个问题,做出来这个问题,小Jerry就能获得糟老头子Tom
赞助的补给!!!只有伟大的团队可以帮得了Jerry,并将获得荣耀之光,你们能帮可怜的Jerry获得补给吗?
给定一个正整数n,求至少两个正整数使得他们的lcm(最小公倍数)为n,且这些正数的和最小,输出这个最小和。

在这里插入图片描述
输入格式
第一行一个整数T(1≤T≤1e4)
接下来T行,每行一个正整数n(1≤n≤231−1)

输出格式
输出T行,每行输出”Case #: “(不包括“”),#表示第#组数据,后面跟题目要求的答案,具体输出可参考样例

样例
input
3
12
10
5
output
Case 1: 7
Case 2: 7
Case 3: 6

数学题

题意:

给一个数字n,范围在[1,2^23-1],这个n是一系列数字的最小公倍数,这一系列数字的个数至少为2

例如12,是1和12的最小公倍数,是3和4的最小公倍数,是1,2,3,4,6,12的最小公倍数,是12和12的最小公倍数………………

那么找出一个序列,使他们的和最小,上面的例子中,他们的和分别为13,7,28,24……显然最小和为7

这个题目一开始没有头绪,后来看了一下网上的题解懂了思路

首先假设我们知道了一系列数字a1,a2,a3……an,他们的LCM是n,那么什么时候他们是最优解呢,当他们两两互质的时候

为了方便我们以两个数来说明问题。

a和b的LCM是n,GCD是m,那么n=a/m*b , 它们的和就是sum=a+b;

如果m不为1(即a和b不互质),那么我们为什么不优化一下,将a变为a=a/m呢?,改变后a和b的LCM依然是n,但是他们的和显然减少了

所以我们得到最重要的一个性质,要想a1,a2,a3……an的和最小,要保证他们两两互质,只要存在不互质的两个数,就一定可以近一步优化

那我们怎么保证两两互质呢?方法其实很简单,直接分解质因子

例如24=222*3 , 只能分解为8和3,因为这里有3个2,这3个2必须在一起,如果分开了这3个2,这出现有两个数会有一个公共的质因子2,并且会使这两个数的LCM不是24

再例如72=22233,只能分为8和9,因为3个2和2个3都不能分开,他们必须在一次

所以,我们将一个数n分解为质因子后,顺便做一个处理,在除干净一个质因子的同时,将他们乘起来作为一个因子,处理完后会得到多个因子,他们之间同样满足两两互质的性质

然后是进一步的分析

例如264600=82725*49 , 只是由3个2,3个3,2个5,2个7,处理后得到的因子,那么8,27,25,49的LCM是264600,并且两两互质,他们还要不要处理呢?不需要了,直接将他们加起来就是我们要的答案!为什么呢?可以将8,27,25,49这些数字乘起来,无论怎样乘都好,最后得到的数字它们的LCM依然是n,但是乘起来再相加显然比直接相加要大得多!

所以我们已经得到了这个问题的解法

1.将一个数分解成质因子,将相同的因子乘起来作为一个处理后的因子

2.将处理后得到的多个因子直接相加就是答案

3.因为题目说只要需要两个数字,所以对于1和素数我们需要小心。对于素数,我们只能分解出一个因子就它自己,对于1一个因子都分解不出来(我们不把1当做因子),他们的答案都是n+1,因为只有1和n的LCM是n

这是这道题的思路以及证明,代码如下:
(在本次比赛中时间限制为2s,所以超时)(应该采用大数据离线打表法,暂且不会)

#include <bits/stdc++.h>


using namespace std;
const int maxx=1e3+7;
typedef long long ll;
#define INF 0x3f3f3f3f
int main()
{
    int t;
    scanf("%d",&t);
    int cas=1;
    while(t--)
    {
        ll n;
        scanf("%lld",&n);
        ll ans=0,cot=0;
        for(int i=2;i*i<=n;i++)
        {
            ll t=1;
            if(n%i==0)
            {
                cot++;
                while(n%i==0)
                {
                    t*=i;
                    n/=i;
                }
                ans+=t;
            }
        }
        if(!cot) ans=n+1;
        else if(cot==1||n!=1) ans+=n;
        printf("Case %d: %lld\n",cas++,ans);
    }
    return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值