XDOJ-1003

题目:最喜欢的数字

描述:

zyf最喜欢的数字是1!所以他经常会使用一些手段,把一些非1的数字变 成1,并为此得意不已。他会且仅会的两种手段是:
1.把某个数m除以某个质数p——当然p必须能整除这个数,即m=m/p
2.把某个数m减1,即m=m-1
有一天他突发奇想,想把[a,b]区间中所有的数一个一个地变成1,这是一个巨大的无聊的工程,所以他想知道他最少得花多少操作才能达到目 的。

输入:

输入包含多组数据(1000组数据),EOF结束。
  每组数据以两个整数开头:a,b(0<a<=b<=100000),意义如题意描述。

输出:

每组数据输出一行,最少操作数。

样例:

2 3
3 5
11 12

样例输出:

2
4
3

 

解题思路:

数字num的最小操作数:

如果num是质数,op(num)=1;

否则,op(num) = min{op(num), op(num/m1),op(num/m2)...},其中m1,m2....是num的质因子。

这可以看作一个动态规划的问题来求解,其中两个核心的问题:(1)如何求得小于M的所有质数(2)如何求解num质因子分解。这两个问题明白后,静心思考一下,可以把个问题的解法描述为:

(1)求出小于100000的所有质数p,并提前标记op(p) = 1。

(2)从小到大计算每个数num的最小操作数。

        a. 如果op(num-1)+1<op(num),则更新op(num)

        b.利用op(num)和之前求得的质数,更新所有num×p<100000的op[num*p]值。这一步是相对不容易想到的,之前用常规的方式总是超时。

(3)根据求得的op数组,和输入的a、b,求和op[a]~op[b]。

 

#include <iostream>

using namespace std;


void cacPrime(bool *isPrime,int* primes,int* minOp,int N)
{
   isPrime[1] = false;
   for(int i=2;i<N;++i)
        isPrime[i] = true;
   int k = 1;
   minOp[1] = 0;
   for(int i=2;i<N;++i)
   {
       if(isPrime[i])
       {
           minOp[i] = 1;
           for(int j=2;i*j<N;++j)
                isPrime[i*j] = false;
            primes[k++] = i;
       }
       else
            minOp[i] = 0;

   }
   for(int i=2;i<N;++i)
   {
       if(minOp[i-1]+1<minOp[i])
            minOp[i] = minOp[i-1]+1;
       for(int j=1;j<k&&i*primes[j]<N;++j)
       {
           if(minOp[i]+1<minOp[i*primes[j]]||minOp[i*primes[j]]==0)
                minOp[i*primes[j]] = minOp[i]+1;
       }

   }
}

int main()
{
    int a,b;
    int N = 100001;
    int minOp[N];//最小操作数
    bool isPrime[N];//表示数是不是素数
    int primes[N/2];
    cacPrime(isPrime,primes,minOp,N);

    while(cin>>a>>b)
    {
        int result = 0;
        for(int i=a;i<=b;++i)
            result += minOp[i];
        cout<<result<<endl;
    }
    return 0;
}

 

最后欢迎大家访问我的个人网站: 1024s

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
xdoj application是一个用于竞技编程训练和练习的平台。 首先,xdoj application提供了大量的编程题库,涵盖了各种语言和难度级别。这些题目既有经典的算法问题,也有实际应用的编程挑战,可以满足不同水平的程序员的需求。通过解题,程序员可以提高自己的编程能力和算法思维。 其次,xdoj application提供了在线的程序编译和运行环境。程序员可以直接在平台上编写和测试自己的代码,无需安装任何开发工具。这大大降低了环境配置的难度,让程序员能够更加专注于解决问题本身。 另外,xdoj application还支持代码提交和评测功能。程序员可以将自己的代码提交到平台上,系统会对其进行自动评测,并给出相应的反馈和评分。这样,程序员可以及时了解自己的代码是否正确并进行相应的优化。同时,平台上也会展示其他程序员的提交和评测结果,让程序员能够学习和借鉴其他人的解决思路和代码实现。 最后,xdoj application还提供了练习和比赛等功能。程序员可以选择不同类型的练习模式,根据自己的需求进行编程训练。同时,平台也会定期举办编程比赛,供程序员们进行交流和竞争。这些比赛既能检验程序员的编程水平,也能提供一种实战的机会,加深对编程知识的理解和应用。 总之,xdoj application是一个功能齐全且用户友好的竞技编程平台,能够帮助程序员提高编程能力、锻炼算法思维,并提供交流、学习和竞争的机会。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值