C++ 用循环链表解决约瑟夫环问题

约瑟夫环问题

已知 n 个人(n>=1)围坐一圆桌周围,从 1 开始顺序编号,从序号为 1 的人开始报数,顺时针数到 m 的那个人出列。下一个人又从 1 开始报数,数到m 的那个人又出列。依此规则重复下去,直到所有人全部出列。请问最后一个出列的人的初始编号。

要求

输入人数 n,所报数 m,输出最后一个人的初始编号。

解决思路

  • 首先因为是圆桌问题,使用链表解决的话需要构建循环链表。
  • 接着是出列问题,这里我的设计思路是将指向链表的指针移动到需要出列的人的位置,然后根据正常的链表删除进行操作即可。
  • 最后是类的设计,可以将 head 、 m、 n、链表构建、链表删除等整合到一起进行封装。

代码

  1 // 6.cpp -- using gcc compiler
  2 #include <iostream>
  3 using namespace std;
  4
  5 typedef struct List
  6 {
  7     int num;
  8     struct List *next;
  9 }*pList;
 10
 11 class Josephus
 12 {
 13     public:
 14         Josephus() {}
 15         Josephus(int number, int mes):
 16             n(number),
 17             m(mes){}
 18         void set();
 19         void creat();
 20         void del();
 21     private:
 22         pList head;
 23         int n, m, tmp_n;
 24 };
 25
 26 void Josephus::set()
 27 {
 28     cout << "please input the number of the people: ";
 29     cin >> n;
 30     tmp_n = n;
 31     cout << "please input the m: ";
 32     cin >> m;
 33 }
 34
 35 void Josephus::creat()
 36 {
 37     pList p1, p2;
 38     pList p = new List;
 39     n += 1;
 40     p -> num = 1;
 41     p2 = head = p;
 42     for(int i = 2; i < n; i++)
 43     {
 44         p = new List;
 45         p -> num = i;
 46
 47         p1 = p2;
 48         p2 = p;
 49         p1 -> next = p2;
 50     }
 51
 52     p2 -> next = head;
 53     // output each number of the circle list‘s members.
 54     // and it should be: 1, 2, ..., n
 55     p = head;
 56     cout << "Now, the \"num\" member of the list is: " << endl;
 57     while(n--)
 58     {
 59         if(n == 0)
 60             cout << p-> num << ".";
 61         else
 62         {
 63             cout << p->num << ", ";
 64             p = p -> next;
 65         }
 66     }
 67 }
 68
 69 void Josephus::del()
 70 {
 71     pList p1 = NULL;
 72     pList p2 = head;
 73
 74     n = tmp_n + 1;
 75     while(n--)
 76     {
 77         int s = m - 1;
 78         while(s--)
 79         {
 80             p1 = p2;
 81             p2 = p2 -> next;
 82         }
 83
 84         if(n == 0)
 85         {
 86             p2 = p2 -> next;
 87             p1 -> next = NULL;
 88             cout << "The result is: " << p2 -> num << endl;
 89         }
 90         else
 91         {
 92             p2 = p2 -> next;
 93             p1 -> next = p2;
 94         }
 95     }
 96 }
 97
 98 int main()
 99 {
100     Josephus t;
101     t.set();
102     t.creat();
103     cout << endl;
104     t.del();
105     return 0;
106 }

代码解析

5-9行定义了链表的结构,其中 num就是初始编号

23行的 tmp_n 负责记录 n 的初始值

75行开始进行循环,共循环 n+1 次,每次都将 p2 移动到需要被排除的人的位置。

n=0时为最后一次循环,此时p1、p2都指向最后一个人,输出 num。

n!=0则去掉p2指向的节点,开始下一次循环。

解决约瑟夫环问题的关键在于去掉计数为 m 的那个人,并从他之后开始重新计数。

而以上的解决方案巧妙的利用了循环链表来解决问题。因为对链表的了解不算深入,所以暂时还想不出更好的解决方案。

总结

解决链表问题的关键在于对链表的构建以及其他操作,而链表的问题只靠想象是不可能的解决的,最好还是自己动手画图,这样一切都会变得清晰明了,问题也会迎刃而解。

时间: 2024-08-03 15:30:31

C++ 用循环链表解决约瑟夫环问题的相关文章

C++循环链表解决约瑟夫环问题

约瑟夫环问题可以简单的使用数组的方式实现,但是现在我使用循环链表的方法来实现,因为上午看到一道面试题规定使用循环链表解决约瑟夫环问题. 什么是约瑟夫环? “约瑟夫环是一个数学的应用问题:已知n个人(以编号1,2,3...n分别表示)围坐在一张圆桌周围.从编号为k的人开始报数,数到m的那个人出列:他的下一个人又从1开始报数,数到m的那个人又出列:依此规律重复下去,直到圆桌周围的人全部出列.”(百度百科中的解决办法列出了很多,可以看到循环链表并不是最简单的方法) 这道面试题考察了循环链表的“创建”,

用循环链表解决约瑟夫环的问题

约瑟夫环问题简介 约瑟夫环问题的原来描述为,设有编号为1,2,--,n的n(n>0)个人围成一个圈,从第1个人开始报数,报到m时停止报数,报m的人出圈,再从他的下一个人起重新报数,报到m时停止报数,报m的出圈,--,如此下去,直到所有人全部出圈为止.当任意给定n和m后,设计算法求n个人出圈的次序.  稍微简化一下. 问题描述:n个人(编号0~(n-1)),从0开始报数,报到(m-1)的退出,剩下的人继续从0开始报数.求胜利者的编号. 解题思路 将每个人的编号作为结点值,因为报数是循环着来的,故可

用循环链表解决约瑟夫环问题

约瑟夫环是一个数学的应用问题:已知n个人(以编号1,2,3...n分别表示)围坐在一张圆桌周围.从编号为k的人开始报数,数到m的那个人出列:他的下一个人又从1开始报数,数到m的那个人又出列:依此规律重复下去,直到圆桌周围的人全部出列. 下面我们用循环列表模拟这个过程: 1 //节点定义与单链表相同,在此省略 2 //use cyclical linked list to solve josephus problem 3 template <typename Type> class Linked

golang数据结构之用循环链表解决约瑟夫环问题

josephu.go package link import ( "fmt" ) type Kid struct { ID int next *Kid } func AddKid(num int) *Kid { first := &Kid{} cur := &Kid{} if num < 1 { fmt.Println("不合法") return first } for i := 1; i <= num; i++ { kid := &am

C语言用数组解决约瑟夫环问题

       在罗马人占领乔塔帕特后,39 个犹太人与约瑟夫及他的朋友躲到一个洞中,大家决定宁愿自杀也不要被敌人抓到,于是确定了一个自杀方式,41个人排成一个圆圈,由第1个人开始报数,每报数到第3人该人就必须自杀,然后再由下一个重新报数,直到所有人都自杀身亡为止.然而约瑟夫和他的朋友并不想死去,那么他应该怎样安排他和他的朋友的位置,才能逃脱这场死亡游戏呢?         用C语言解决约瑟夫环问题的最佳方式是采用循环链表,但使用数组同样也可以解决瑟夫环的问题.采用循环链表的方法,以后详述.本节主

javascript中使用循环链表实现约瑟夫环问题

1.问题 2.代码实现 /** * 使用循环链表实现解决约瑟夫环问题 * */ //链表节点 function Node(element){ this.element = element; this.next = null; } //定义链表类 function LList(){ this.head = new Node("head"); this.head.next = this.head; this.find = find; this.insert = insert; this.f

循环链表解决约瑟夫问题

训练一下尾插法和循环链表的使用. //循环链表解决约瑟夫问题 #include <stdio.h> #include <stdlib.h> typedef struct CycleLinkList { int data; struct CycleLinkList * next; }cycleLinkList; cycleLinkList * h, * r, * s; // 头指针.尾指针 int main() { int i, j; int n, num; cycleLinkLis

php解决约瑟夫环的问题

php里面解决约瑟夫环还是比较方面的,但是下面的方法太费空间 <?php class SelectKing{ private $m;//幅度 private $n;//总数 public function __construct($m,$n){ $this->m = $m; $this->n = $n; } public function getKing(){ $mokeys = range(1, $this->n); $tmp = 0; while(count($mokeys)&

(java描述)关于链表的代码-----单双、循环链表、约瑟夫环、多项式相加

将链表头尾倒置 将几个链表合并成一个新的链表,将链表中重复的节点去掉,并按大小排序 双向循环链表 单向循环链表(约瑟夫循环) 多项式相加 程序源代码 单链表.单向循环链表结点类 package javab; public class Node { int data; Node next; public Node(int data){ this.data=data; } } 第一题代码: package javab; import java.util.Scanner; public class I