20210402-算法学习-链表(Linked List)-单项环形链表(Josephu问题)
一.单项环形链表应用场景
1.Josephu(约瑟夫,约瑟夫环) 问题:
Josephu 问题:设编号为1,2,... ..n的n个人围坐一圈,约定编号为k(1<=k<=n) 的人从1开始报数,数到m的那个人出列,他的下一为又从1开始报数,数到m的那个人又出列,以此类推,直到所有人出列为止,由此产生一个出队编号的序列
提示:用一个不带头的节点的环形链表来处理Josephu 问题:先构成一个有n个节点的单循环链表,然后由k节点起从1开始数,计到m时,把对应的节点从链表中删除,然后从被删除的节点的下一个开始又从1开始数,直到最后一个节点从链表中删除,算法结束
2.单项环形链表介绍:
     图1.1
图1.1
2.1.约瑟夫问题示意图(图1.2):
  Josephu  问题为:设编号为1,2,… n的n个人围坐一圈,约定编号为k(1<=k<=n)的人从1开始报数,数到m 的那个人出列,它的下一位又从1开始报数,数到m的那个人又出列,依次类推,直到所有人出列为止,由此产生一个出队编号的序列。
  n = 5 , 即有5个人 
k = 1, 从第一个人开始报数
m = 2, 数2下
k = 1, 从第一个人开始报数
m = 2, 数2下
    图1.2
图1.2
构建一个单向的环形链表思路:
    1. 先创建第一个节点, 让 first 指向该节点,并形成环形
2. 后面当我们每创建一个新的节点,就把该节点,加入到已有的环形链表中即可.
2. 后面当我们每创建一个新的节点,就把该节点,加入到已有的环形链表中即可.
  遍历环形链表:
1. 先让一个辅助指针(变量) curBoy,指向first节点
2. 然后通过一个while循环遍历 该环形链表即可 curBoy.next == first 结束
1. 先让一个辅助指针(变量) curBoy,指向first节点
2. 然后通过一个while循环遍历 该环形链表即可 curBoy.next == first 结束
 图1.3
图1.3
二. 代码:
package com.atAlgorithmTest;
/**
 * @Author: lisongtao
 * @Date: 2021/4/2 10:53
 */
/**
 * @ClassName Josephu
 * @Description: 约瑟夫问题示
 * @Author DELL
 * @Date 2021/04/02 10:53
 **/
public class Josephu {
    public static void main(String[] args) {
        CircleSingleLinkList circleSingleLinkList = new CircleSingleLinkList();
        circleSingleLinkList.add(5);
        circleSingleLinkList.list();
        circleSingleLinkList.count(1,2,5);
    }
}
//创建一个环形的单项链表
class CircleSingleLinkList{
    //创建一个first 节点,当前没有编号
    private Boy head = null;
    //添加一个小孩节点,构成一个环形链表
    public void add(int no){
        if (no<1){
            System.out.println("no值不正确");
            return;
        }
        Boy temp = null;
        for (int i = 1; i <=no; i++) {
            Boy boy = new Boy(i);
            if (i == 1){
                head = boy;
                head.setNext(head);
                temp = head;
            }else{
                temp.setNext(boy);
                boy.setNext(head);
                temp = boy;
            }
        }
    }
    //遍历当前环形链表
    public void list(){
        if (head == null){
            System.out.println("链表为空");
            return;
        }
        Boy temp = head;
        while (true){
            System.out.printf("小孩编号%d\n",temp.getNo());
            if (temp.getNext() == head){
                break;
            }
            temp = temp.getNext();
        }
    }
    //根据输入,计算出,出圈的顺序
    public void count(int start,int count,int no){
        //对数据进行校验
        if (head == null || start<1 || start >no){
            System.out.println("输入的参数有误,请重新输入");
            return;
        }
        Boy temp = head;
        while (true){
            if (temp.getNext() == head){
                break;
            }
            temp = temp.getNext();
        }
        for (int i = 0; i <start-1 ; i++) {
            head = head.getNext();
            temp = temp.getNext();
        }
        while (true){
            if (temp == head){//说明当前圈中只有一个节点
                break;
            }
            for (int i = 0; i <count-1 ; i++) {
                head = head.getNext();
                temp = temp.getNext();
            }
            System.out.printf("小孩%d 出圈\n",head.getNo());
            head = head.getNext();
            temp.setNext(head);
        }
        System.out.printf("最后留在圈中的小孩的编号%d\n",head.getNo());
    }
}
//创建一个Boy类,表示一个节点
class Boy{
    private int no;//编号
    private Boy next;//指向下一个节点,默认null
    public Boy(int no){
        this.no = no;
    }
    public int getNo() {
        return no;
    }
    public void setNo(int no) {
        this.no = no;
    }
    public Boy getNext() {
        return next;
    }
    public void setNext(Boy next) {
        this.next = next;
    }
}
 
                    
                
 
                
            
         浙公网安备 33010602011771号
浙公网安备 33010602011771号