问题描述:

已知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++)的更多相关文章

  1. 约瑟夫环问题详解(java版)

    1 什么是约瑟夫环问题? 约瑟夫,是一个古犹太人,曾经在一次罗马叛乱中担任将军,后来战败,他和朋友及另外39个人躲在一口井里,但还是被发现了.罗马人表示只要投降就不死,约瑟夫想投降,可是其他人坚决不同 ...

  2. 【OSPF】防环机制详解

    我们在提到OSPF的时候,时常喜欢说的一句话就是,OSPF能够计算出无环的路由,那么OSPF究竟是如何规避路由环路的呢?OSPF与距离矢量路由协议不同,运行OSPF的路由器之间交互并不是路由信息,而是 ...

  3. C语言解决约瑟夫问题详解的代码

    将开发过程中比较重要的一些内容做个收藏,下面的内容是关于C语言解决约瑟夫问题详解的内容,希望能对码农有帮助. #pragma once #include<vector> class PRO ...

  4. 1-25-循环控制符break、continue和函数详解

    大纲: 1-for循环补充 1-1-for循环实战---类C格式应用 2-break.continue循环控制符 2-1实战:帮助理解break.continue作用 3-函数详解 3-1.脚本文件中 ...

  5. 详解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 ...

  6. C++版 - 剑指Offer 面试题45:圆圈中最后剩下的数字(约瑟夫环问题,ZOJ 1088:System Overload类似)题解

    剑指Offer 面试题45:圆圈中最后剩下的数字(约瑟夫环问题) 原书题目:0, 1, - , n-1 这n个数字排成一个圈圈,从数字0开始每次从圆圏里删除第m个数字.求出这个圈圈里剩下的最后一个数字 ...

  7. Java实现 LeetCode 面试题62. 圆圈中最后剩下的数字(约瑟夫环)

    面试题62. 圆圈中最后剩下的数字 0,1,n-1这n个数字排成一个圆圈,从数字0开始,每次从这个圆圈里删除第m个数字.求出这个圆圈里剩下的最后一个数字. 例如,0.1.2.3.4这5个数字组成一个圆 ...

  8. Drawable实战解析:Android XML shape 标签使用详解(apk瘦身,减少内存好帮手)

    Android XML shape 标签使用详解   一个android开发者肯定懂得使用 xml 定义一个 Drawable,比如定义一个 rect 或者 circle 作为一个 View 的背景. ...

  9. 详解javascript的类

    前言 生活有度,人生添寿. 原文地址:详解javascript的类 博主博客地址:Damonare的个人博客 Javascript从当初的一个"弹窗语言",一步步发展成为现在前后端 ...

随机推荐

  1. CreateWindow() -- 创建普通的窗口

    (1)函数原型 1 HWND CreateWindow( 2 LPCTSTR lpClassName, //pointer to register class name 3 LPCTSTR lpWin ...

  2. 【LeetCode】面试题62. 圆圈中最后剩下的数字

    题目:面试题62. 圆圈中最后剩下的数字 这题很有意思,也很巧妙,故记录下来. 官方题解思路,是约瑟夫环的数学解法: 我们将上述问题建模为函数 f(n, m),该函数的返回值为最终留下的元素的序号. ...

  3. element UI表格行高、padding等设置报错问题

    element UI里面表格的行高需要自己调整高度和设置padding,直接写style是不行的,里面有 : 1.row-style (行的 style) 2.header-row-styl   (表 ...

  4. 数据结构和算法学习笔记十五:多路查找树(B树)

    一.概念 1.多路查找树(multi-way search tree):所谓多路,即是指每个节点中存储的数据可以是多个,每个节点的子节点数也可以多于两个.使用多路查找树的意义在于有效降低树的深度,从而 ...

  5. 透视RPC协议:SOFA-BOLT协议源码分析

    前提 最近在看Netty相关的资料,刚好SOFA-BOLT是一个比较成熟的Netty自定义协议栈实现,于是决定研读SOFA-BOLT的源码,详细分析其协议的组成,简单分析其客户端和服务端的源码实现. ...

  6. 8.7考试总结(NOIP模拟)[Smooth·Six·Walker]

    前言 踩了挺多以前没踩过的坑... T1 一开始是打了一个 60pts 的 DFS ,在与暴力拍了几组数据保证正确性之后, 突然想到 BFS 可能会更快一些,然后就又码了一个 BFS,又和 DFS 拍 ...

  7. ContentObserver 内容观察者作用及特点

    eg: 1.定义Uri public static Uri KEY_BROWSER_URI = Uri.parse("content://com.android.browser.provid ...

  8. Java互联网架构师系统进阶课程 (一)【享学】

    2.线程的并发工具类 Fork-Join 什么是分而治之? 规模为N的问题,N<阈值,直接解决,N>阈值,将N分解为K个小规模子问题,子问题互相对立,与原问题形式相同,将子问题的解合并得到 ...

  9. ceph介绍和安装

    目录 1.Ceph简介 2.Ceph的特点 3.Ceph的缺点 4.架构与组件 4.1.组件介绍 4.2.存储过程 5.部署 5.1 设置主机名.配置时间同步 5.2 配置添加清华源 5.3 初始化c ...

  10. Package Repository for Ubuntu Offline Installation

    For example, we need install "unzip" command on a Ubuntu machine without Internet. On a on ...