北大ACM(1012Joseph)代码

/*
	Memory 92K Time 47MS
*/
#include <stdio.h>
#include <stdlib.h>


/*
	解约瑟夫问题
	假如存在n个人 其中有k个坏人k个好人 每隔m个月就杀掉一个人,你需要保证所有的坏人都在好人前面被杀掉
	并且最后剩下一个好人
	现在要求你输入 k  其中前一个k的数量表示好人的数量 后一个k表示坏人的数量
	然后输出与k对应的最小m
*/
/*
	计算出 约瑟夫问题的公式为 
	(m + start -1) % (n - killCount)			
	(轮数m + 开始下标start - 1) % (数字总数n - 以及被kill的数字数量killCount)			
	获取下一轮被kill的数字下标,其中下标不为实际上的下标 而是跳过被kill的number 的下标
	也就是当前剩余数的下标
*/

#define MAX_SIZE 14
int calc(int m, int n);


int main()
{
	int gResult[MAX_SIZE];		//保存题目要求的1-13个数字对应的结果
	int m, k;				


	for (k = 1; k < MAX_SIZE; ++k)
	{
		m = k + 1;					//对于给定一个数字 它的周期数不会少于自身+1
		while (1)
		{
			if (calc(m, k * 2))
			{
				gResult[k - 1] = m;
				break;
			}
			else if (calc(m + 1, k * 2))
			{
				gResult[k - 1] = m + 1;
				break;
			}
			m += k + 1;				//每次将轮数m递增 增长率至少为 k
		}
	}

	while (scanf_s("%d", &k) && k)
	{
		printf_s("%d\n", gResult[k - 1]);
	}

	return 0;
}
int calc(int m, int n)			//对于给定的m轮数来计算出是否能够满足数字总数n的要求
{
	int killCount = 0;
	int start = 0;		//每次开始的下标	

	///循环进行检测  如果出现了下标小于 n / 2表示该m为不可行方案 直接返回假
	while (n - killCount > n / 2)
	{
		start = (m + start - 1) % (n - killCount);
		if (start < n / 2)	return 0;
		killCount++;
	}
	return 1;
}
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

David_TD

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值