约瑟夫问题(Joseph)

约瑟夫问题(Joseph)


设有N名同学手拉手围成一圈,自1、2、3、......开始编号,现从1号开始连续数数,
每数到M将此同学从圈中拉走, 求最后被拉走的同学的编号

 
输入
两个正整数,分别为N和M。0<N<=100 ;0<M<65535

 
输出
一个正整数,为最后被拉走的同学的编号,最后输出回车

 
输入样例
5 3
 
输出样例
4
 
提示
 
来源









#include <stdio.h>

int main()
{
    // 假设k=l为报到计数单位量
    int i,k,m,n,num[50],*p,l;
    scanf("%d %d",&n,&l);
    p=num;

    for(i=0;i<n;i++)
        *(p+i)=i+1;//以1至n为序给每个人编号
    i=0;//i为每次循环时计数变量
    k=0;//k为按1,2,3...l报数时的计数变量
    m=0;//m为退出人数
    while (m<n-1)//当退出人数比n-1少时执行循环体
    {
        if(*(p+i)!=0) k++;
        if (k==l)
        {
            //可以将每次出局的人编号依次输出来
            //printf("出局人序号:%d\n",*(p+i));
         *(p+i)=0;//将退出的人的编号置为0
        k=0;//k报到l后,重置为0
        m++;//退出的人数+1
        }
        i++;
        if (i==n) i=0;//报数到尾后,i恢复为0
    }
    while (*p==0) p++;//如果p所指向的值等于0.那么就执行p++让它指向下一个元素,直到不为0.
    //printf("最后留下的人的编号是:%d\n",*p);//经过上面的循环后,*p的指向的编号就是最后留下的人
    printf("%d\n",*p);
    return 1;
}



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值