约瑟夫环问题详解 (c++)
问题描述:
已知n个人(以编号0,2,3...n-1分别表示)围坐在一起。从编号为0的人开始报数,数到k的那个人出列;他的下一个人又从1开始报数,数到k的那个人又出列;依此规律重复下去,直到圆桌周围的人全部出列,最后一个出列的人为胜利者。求胜利者编号.
历史背景:
Wikipedia: 这个问题是以弗拉维奥·约瑟夫斯命名的,它是1世纪的一名犹太历史学家。
他在自己的日记中写道,他和他的40个战友被罗马军队包围在洞中。
他们讨论是自杀还是被俘,最终决定自杀,并以抽签的方式决定谁杀掉谁。约瑟夫斯和另外一个人是最后两个留下的人。
约瑟夫斯说服了那个人,他们将向罗马军队投降,不再自杀。约瑟夫斯把他的存活归因于运气或天意,他不知道是哪一个.
问题分析:解决该问题有两种思路,第一种:通过建立循环链表来模拟这个过程
第二种:通过递归方式(数学归纳将问题转化为数学问题)
由于递归方式,代码更简洁,下面首先以递归方式来解决问题
——————递归实现:
例如 对 m= 10,k=3
0 1 2 3 4 5 6 7 8 9 (*)
0 1 3 4 5 6 7 8 9 (* 循环下去)
转化为: 3 4 5 6 7 8 9 0 1 (* 循环下去)
0 1 2 3 4 5 6 7 8
m=10,k=3 去掉一个元素之后,变成了一个m=9,k=3的约瑟夫环问题
并且有如下关系
3 = (0+3)%10 4 = (1+3)%10 ... 0 = (3+7)%10
即 3 = (0+k)% m 4 = (1+k) % m ... 0 = (3+k) % m
m =10,k =3 设约瑟夫环最后一个出列的人为 Joseph(10,3),那么存在如下关系
Joseph(10,3) = (Joseph(9,3)+k) %m;
...
Joseph(n,k) = (Joseph(n-1,k)+k) % n (n>1);
C++实现如下:
递归方法一:
1 int Joseph(int m,int k)
2 {
3 if(m<=0||k<=0)
4 {
5 cout<<"error!"<<endl;
6 return -1;
7 }else
8 {
9 if(m==1)
10 {
11 return 0;
12 }else
13 {
14 return ((Joseph(m-1,k)+k)%m);
15 }
16 }
17 }
递归方法二:如果输出整个出队的顺序
int Joseph(int m,int k,int i)
{
if(m<=0||k<=0||m<i)
{
cout<<"error"<<endl;
return -1; }else
{
if(i==1)
{
return (m+k-1)%m;
}else
{
return ((Joseph(m-1,k,i-1)+k)%m);
}
}
}
程序运行结果如下:
int main()
{
cout<<"递归方法一"<<endl;
cout << Joseph(6,3) << endl;
cout<<"递归方法二"<<endl;
for(int i=1;i<=6;i++)
{
cout<<Joseph(6,3,i)<<endl;
}
getchar();
return 0;
}
结果:

——————循环链表实现:
建立节点数据结构:循环链表
struct Node
{
int data;
Node * next;
}; struct LinkedList
{
Node *pHead;
Node *pTail;
int len;
};
建立循环链表
//建立个节点
Node * GetNode(int i)
{
Node * p = (Node *)malloc(sizeof(Node));
if(p!=NULL&&i>=0)
{
p->data = i;
p->next = NULL;
return p; }else
{ cout<<"error"<<endl;
exit(-1);
return NULL;
}
} //建立链表
LinkedList* CreateLinkedList(int i)
{
Node* node = GetNode(0);
LinkedList *head = (LinkedList*)malloc(sizeof(LinkedList));
if(head == NULL)
{
cout<<"CreateLinkedList:memory error"<<endl;
exit(-1);
return NULL;
}
if(i<=0)
{
cout<<"CreateLinkedList: error"<<endl;
exit(-1);
return NULL;
} head->pHead = node;
head->pTail = node;
head->len = 1;
if(i==1)
{
node->next = node;
}else
{
Node *p = head->pHead;
for(int j=1;j<=i-1;j++)
{
Node* node = GetNode(j);
node->data = j;
p->next = node;
p=p->next;
head->len++;
}
head->pTail = p;
p->next = head->pHead;
}
return head; }
删除节点:
//删除节点
void RemoveNode(LinkedList*head,Node *deleNode)
{
Node* p = head->pHead;
cout<<deleNode->data<<endl;
if(p!=NULL){
if(head->len>1){
do
{
if(p->data==deleNode->data)
{
if(p==head->pHead)
{
head->pHead = p->next;
}
if(p==head->pTail)
{
head->pTail = p->next;
}
p->data = p->next->data;
p->next = p->next->next;
head->len--;
return; }else{
p=p->next;
}
}while(p!=head->pHead);
}else
{
cout<<"error";
exit(-1);
return;
}
}else
{
return;
}
}
约瑟夫模拟:
int Joseph(int m,int k)
{
if(m<=0||k<=0)
{
cout<<"error:input"<<endl;
return -1;
}
LinkedList* list = CreateLinkedList(m);
//Print_List(list);
Node * p = list->pHead; for(int i=1;i<=k;i++)
{
if(list->len ==1)
{
return p->data;
}
if(i==k)
{
RemoveNode(list,p);
i = 1;
}
p=p->next;
}
return 0; }
程序运行结果如下:
int _tmain(int argc, _TCHAR* argv[])
{ cout<<"循环列表"<<endl;
cout<<Joseph(6,3)<<endl;
getchar();
return 0;
}

约瑟夫环问题详解 (c++)的更多相关文章
- 约瑟夫环问题详解(java版)
1 什么是约瑟夫环问题? 约瑟夫,是一个古犹太人,曾经在一次罗马叛乱中担任将军,后来战败,他和朋友及另外39个人躲在一口井里,但还是被发现了.罗马人表示只要投降就不死,约瑟夫想投降,可是其他人坚决不同 ...
- 【OSPF】防环机制详解
我们在提到OSPF的时候,时常喜欢说的一句话就是,OSPF能够计算出无环的路由,那么OSPF究竟是如何规避路由环路的呢?OSPF与距离矢量路由协议不同,运行OSPF的路由器之间交互并不是路由信息,而是 ...
- C语言解决约瑟夫问题详解的代码
将开发过程中比较重要的一些内容做个收藏,下面的内容是关于C语言解决约瑟夫问题详解的内容,希望能对码农有帮助. #pragma once #include<vector> class PRO ...
- 1-25-循环控制符break、continue和函数详解
大纲: 1-for循环补充 1-1-for循环实战---类C格式应用 2-break.continue循环控制符 2-1实战:帮助理解break.continue作用 3-函数详解 3-1.脚本文件中 ...
- 详解OJ(Online Judge)中PHP代码的提交方法及要点【举例:ZOJ 1001 (A + B Problem)】
详解OJ(Online Judge)中PHP代码的提交方法及要点 Introduction of How to submit PHP code to Online Judge Systems Int ...
- C++版 - 剑指Offer 面试题45:圆圈中最后剩下的数字(约瑟夫环问题,ZOJ 1088:System Overload类似)题解
剑指Offer 面试题45:圆圈中最后剩下的数字(约瑟夫环问题) 原书题目:0, 1, - , n-1 这n个数字排成一个圈圈,从数字0开始每次从圆圏里删除第m个数字.求出这个圈圈里剩下的最后一个数字 ...
- Java实现 LeetCode 面试题62. 圆圈中最后剩下的数字(约瑟夫环)
面试题62. 圆圈中最后剩下的数字 0,1,n-1这n个数字排成一个圆圈,从数字0开始,每次从这个圆圈里删除第m个数字.求出这个圆圈里剩下的最后一个数字. 例如,0.1.2.3.4这5个数字组成一个圆 ...
- Drawable实战解析:Android XML shape 标签使用详解(apk瘦身,减少内存好帮手)
Android XML shape 标签使用详解 一个android开发者肯定懂得使用 xml 定义一个 Drawable,比如定义一个 rect 或者 circle 作为一个 View 的背景. ...
- 详解javascript的类
前言 生活有度,人生添寿. 原文地址:详解javascript的类 博主博客地址:Damonare的个人博客 Javascript从当初的一个"弹窗语言",一步步发展成为现在前后端 ...
随机推荐
- java构造器级简单内存分析
java构造器的使用(基础篇) 构造方法也叫构造器,是创建对象时执行的特殊方法,一般用于初始化新对象的属性. 基本定义语法: 访问控制符 构造方法名([参数列表]){ 方法体 } 注:"访问 ...
- 论文笔记:(2019)LDGCNN : Linked Dynamic Graph CNN-Learning on PointCloud via Linking Hierarchical Features
目录 摘要 一.引言 A.基于视图的方法 B.基于体素的方法 C.基于几何的方法 二.材料 三.方法 A.问题陈述 B.图生成 C.图特征提取 D.变换不变函数 E.LDGCNN架构 F.冻结特征提取 ...
- user-agent随笔
常用的user-agent: 一.pc端的user-agent汇总,各大浏览器 1.safari 5.1 – MAC Mozilla/5.0 (Macintosh; U; Intel Mac OS X ...
- C++1-100之间 7的倍数 带7 打印 敲桌子
1 // 1-100之间 7的倍数 带7 打印 敲桌子 2 #include <iostream> 3 using namespace std; 4 5 int main() 6 { 7 ...
- jmeter永久调为中文
将jmeter调为中文有两种方法,一是在软件设置中切换,二是修改配置文件. 第一种方式是临时的,下次重新打开会变回为英文 第二种方式是永久的,每次打开都会显示自己配置好的语言 第一种方式: 第二种方式 ...
- 揭秘阿里云 RTS SDK 是如何实现直播降低延迟和卡顿
作者:予涛 途坦 这个夏天,没什么能够比一场酣畅淋漓的奥运比赛来的过瘾.但是,在视频平台直播观看比赛也有痛点:"卡顿" 和 "延时".受限于不同地域.复杂的网络 ...
- Linux 内核预备知识:浅析 offsetof 宏以及新手的所思所想
最近一头扎进了 Linux 内核的学习中,对于我这样一个没什么 C 语言基础的新生代 Java 农民工来说实在太痛苦了.Linux 内核的学习,需要的基础知识太多太多了:C 语言.汇编语言.数据结构与 ...
- git根据项目地址使用不同代理服务器
问题 由于公司访问GitHub只能走代理,但是内网gitlab服务器又不能走代理. 因此想找到一种方案,可以支持git自动根据项目地址使用不同代理. 方案 如下所示,可以指定GitHub地址使用指定的 ...
- Redis配置及攻击利用
Redis配置及攻击利用 Redis及其安全配置 Redis介绍 redis默认会绑定在 0.0.0.0:6379,如果没有进行采用相关的策略,比如添加防火墙规则避免其他非信任来源 ip 访问等,这样 ...
- 解决docker删除加载失败的镜像报错
背景: 准备在vulhub复现weblogic反序列化漏洞时报错,环境加载失败准备删除weblogic镜像时报错: unable to delete 7d35c6cd3bcd (must be for ...