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

约瑟夫环是一个数学的应用问题:已知n个人(以编号1,2,3...n分别表示)围坐在一张圆桌周围。从编号为k的人开始报数,数到m的那个人出列;他的下一个人又从1开始报数,数到m的那个人又出列;依此规律重复下去,直到圆桌周围的人全部出列。

下面我们用循环列表模拟这个过程:

 1 //节点定义与单链表相同,在此省略
 2 //use cyclical linked list to solve josephus problem
 3 template <typename Type> class LinkedList {
 4 private:
 5     Node<Type> *head;
 6 public:
 7     LinkedList() {
 8         head = NULL;
 9     }
10
11     ~LinkedList() {
12         if(head==NULL){
13             return;
14         }
15         //head表示尾节点,head->next表示头节点
16         Node<Type> *current_node = head->next;
17         //拆成单链表
18         head->next=NULL;
19         while (current_node != NULL) {
20             Node<Type> *delete_node = current_node;
21             current_node = current_node->next;
22             delete delete_node;
23         }
24     }
25
26     bool insert(Node<Type> *node, int index) {
27         if (head == NULL) {
28             if (index != 0) {
29                 return false;
30             }
31             head = node;
32             //只有一个节点时,next指向自己
33             head->next=head;
34             return true;
35         }
36         if (index == 0) {
37             node->next = head->next;
38             head->next = node;
39             return true;
40         }
41         Node<Type> *current_node = head->next;
42         int count = 0;
43         while (current_node != head && count < index - 1) {
44             current_node = current_node->next;
45             count++;
46         }
47         if (count == index - 1) {
48             node->next = current_node->next;
49             current_node->next = node;
50             //如果加到链表末尾,需更新尾节点
51             if(node==head->next){
52                head=node;
53             }
54             return true;
55         }
56         return false;
57     }
58
59     //删除数到m的元素并按被删顺序输出,最后被输出的即为留到最后的
60     void output_josephus(int m){
61         Node<Type> *current_node=head;
62         //之后不再使用head指针,所以在这里把它指空,防止最后成为野指针,
63         //因为尾节点可能会被删掉
64         head=NULL;
65         //循环条件:剩下不止一个元素
66         while(current_node->next!=current_node){
67             //从尾节点开始数,数到m时恰好是从头数第m-1个元素
68             for(int i=1;i<m;i++){
69                 current_node=current_node->next;
70             }
71             cout<<current_node->next->data<<" ";
72             Node<Type> *delete_node=current_node->next;
73             current_node->next=current_node->next->next;
74             delete delete_node;
75         }
76         //最后剩余的一个节点
77         cout<<current_node->data<<endl;
78         delete current_node;
79     }
80 };
时间: 2024-10-04 10:03:11

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

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

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

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

约瑟夫环问题 已知 n 个人(n>=1)围坐一圆桌周围,从 1 开始顺序编号,从序号为 1 的人开始报数,顺时针数到 m 的那个人出列.下一个人又从 1 开始报数,数到m 的那个人又出列.依此规则重复下去,直到所有人全部出列.请问最后一个出列的人的初始编号. 要求 输入人数 n,所报数 m,输出最后一个人的初始编号. 解决思路 首先因为是圆桌问题,使用链表解决的话需要构建循环链表. 接着是出列问题,这里我的设计思路是将指向链表的指针移动到需要出列的人的位置,然后根据正常的链表删除进行操作即可.

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

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

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