质数与合数

质数与合数

题意:

FFF和GGG正在玩一个质数与合数的游戏
一开始有N个石头
FFF和GGG轮流对这堆石头进行操作,FFF每次选择1到K之间的一个数x,并拿走x个石头,拿走之后剩下的石头数量必须是质数
接着GGG进行同样的操作,但是要求拿走之后剩下的石头数量必须是合数
假设双方都足够聪明,第一个不能操作的人输
以为到这里就结束了?
显然,一开始两人之间有一人知道自己有必胜策略,现在他想要尽可能快的赢,另外一个人则想要尽可能输的慢一点
给你N与K,假设游戏最终持续了X轮,如果FFF赢了,输出X,否则输出-X
注意:1既不是质数也不是合数

题解:

判断输赢很简单,因为每次取数范围是[1,K],而FFF要每次取完都是质数,如果存在两个相邻的质数,差大于K+1,也就是无论取任何数都不可能剩下为质数,那么FFF就输了,反之FFF必胜,因为FFF都能取到,但是游戏到最后时,剩下34,5,GGG都会输,所以FFF必胜
本题麻烦在计算游戏会进行几轮
根据题意,游戏的胜负从一开始就给定了,但是赢的人想赶紧赢,输的人想慢点输。如果是FFF输了,保证GGG步骤最少,呢么GGG每次都走到下一个素数的前一个的位置,(相当于故意恶心FFF,让FFF正好错过自己的质数),FFF就永远不能跳过相差大于k+1的两个点。
就比如当前点是31,K=3,轮到GGG了,现在离30最近的质数是29,GGG就走到28(因为GGG不能走到质数),让FFF错过自己的质数
如果是FFF赢了,他想赶紧结束比赛就会尽可能跳更远的质数,而GGG为了拖慢,每次只会跳一个单位,当然当跳到2,3时GGG就算输了,不能再跳

代码:

懒得写了,官方代码,但是我每句话做了详细注释

#include <bits/stdc++.h>
using namespace std;
const int N=1e7+5;

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

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

int main()
{
    init();
    bool flag=true;
    int n,k,pos,m=0;
    scanf("%d%d",&n,&k);
    if(n<=2) {puts("0");return 0;}
    for(int i=1;prime[i]<=n;i++)
    {
        m++;
        if(prime[i]-prime[i-1]>k+1)//如果有距离大于k+1,FFF就输了 
        {
            flag=false;
        }
    }
    if(n-prime[m]>k) {puts("0");return 0;}//如果第一步FFF都跳不了 
    if(flag)//FFF可以赢
    {
       // puts("9");
        //FFF可以赢,那么每次选k的时候会尽量多选些质数.
        int wz,ans=0;
        while(1)
        {
            wz=n-k;
            int xb=lower_bound(prime,prime+m,wz)-prime;
            //找第一个大于等于wz的质数,也就是离n最远的质数 
            ans++;//FFF操作完毕 
            if(prime[xb]<=3) break;//如果当前点小于等于3,GGG就输掉比赛 
            n=prime[xb]-1;//GGG所跳点 
            ans++;//GGG操作完毕 
        }
        printf("%d\n",ans);
    }
    else//GGG可以赢
    {
       // cout<<prime[pos]<<endl;
        int wz,ans=0;int cnt=m;
        if(n==prime[m])//如果起点就是质数 
        {
            if(n<=prime[m-1]+k) cnt--;//如果第一步FFF可以走的话 
            else 
            {
                puts("0");
                return 0;
            }
        }
        while(1)
        {
            if(n<=prime[cnt]+k)//如果FFF可以走 
            {
                n=prime[cnt];//FFF走一步 
            }
            else break;
            ans++;
            int wz=n-k; 
            int xb=upper_bound(prime,prime+m,wz)-prime;
            //找第一个大于n-k的质数,也就是离n最近的质数 
            n=prime[xb]-1;//GG正好走到prime[xb]-1(也就是素数的前一个位置来恶心FFF) 
            cnt=xb-1;//FFF下一个最近的质数是第xb-1位(因为xb位被GGG给“完美错过”) 
            ans++;
        }
        if(ans) printf("%d\n",-ans);
        else    puts("0");
    }
    return 0;
}
全部评论

相关推荐

断电再接线:1. 简历排版方面,你这内容比较少,一页放完。各模块之间建议用明显的分隔线分开,现在一眼看上去非常乱。教育经历留白太多。项目经历格式不统一。 2. 第一个项目,硬件设计太笼统,硬件架构规划是指板级电路设计还是FPGA逻辑设计?FPGA时序逻辑设计具体指的什么?实现的三个低速协议以及使用协议进行控制时序,是指什么? 3. 第二个项目,我觉得你可以和第一个项目整合一下,合并为一个项目。第二个项目说实话随便买个zynq开发板都有一直petalinux的教程,作为一个独立的项目不合适的,更像是一个小作业。 4. 第三个项目,项目内容这里,其实和环境搭建之类的东西提一嘴就好了,环境准备和编译安装工具链这种东西没多大必要写,实在要写的话可以 说 使用docker 独立sdk环境之类的。你说的这个工具我没用过,我用的比较多的是busybox和buildroot,是基于menuconfig进行配置的,如果scratch也是类似的模式的话,那我觉得这个项目也经不起细推。你可以往内核裁剪那方向靠,我说的这两个工具你也可以看看。 5. 你熟悉这些接口时序的话,你可以进一步去看一下驱动开发,然后面试的时候突出这个作为重点。第三个项目也可以将驱动开发给补充进去。因为单编内核和构建文件系统,其实很多时候是体力劳动。 6. 特长这里,独立成一个荣誉奖项的模块,把你获得的奖学金和竞赛奖项放一起。数模的话,写了国赛,美赛就不用写了。 7. 总的来说可以了,你简历上写的东西你只要都熟悉,找个实习还是没问题的。 8. 嵌入式分为硬件,底层软件和应用软件,看你的经历我建议你往底层靠,多去熟悉常用的通信接口,去看内核和驱动,网络编程这块也可以去了解一下。然后去**刷刷hot100
点赞 评论 收藏
分享
评论
点赞
收藏
分享
牛客网
牛客企业服务