关于单向循环链表的约瑟夫问题(Java实现)

最近在学习链表时,遇到单向循环链表中的约瑟夫问题。在构建循环链表的代码上,我有一点很不理解,遂记录下来。

Josephu问题为:

设编号为1, 2,.. n的n个人围坐- -圈,约定编号为k (1<=k<=n)的人从1开始报数,数到m的那个人出列,它的下一-位又从1开始报数,数到m的那个人又出列,依次类推,直到所有人出列为止,由此产生一个出队编号的序列。

提示:用一个不带头结点的循环链表来处理Josephu问题:先构成一个有n个结点的单循环链表,然后由k结点起从1开始计数,计到m时,对应结点从链表中删除,然后再从被删除结点的下一个结点又从1开始计数,直到最后一个结点从链表中删除算法结束。

构建一个单向的环形链表思路:
1.先创建第一个节点让first指向该节点,并形成环形
2.后面当我们每创建- -个新的节点,就把该节点,加入到已有的环形链表中即可.
遍历环形链表
1.先让一个辅助指针(变量)curBoy,指向frt节点
2.然后通过一-个while循环遍历该环形链表即可curBoy.next == first结束

代码清单:

package lianbiao;
public class Josepfu {
  public static void main(String[]args){
         CircleSingleLinkedList circleSingleLinkedList =new CircleSingleLinkedList();
         circleSingleLinkedList.addBoy(5);//加入5个小孩节点
         circleSingleLinkedList.showBoy();
         circleSingleLinkedList.countBoy(1, 2, 5);
         }
}
 //创建一个环形的单向链表
 class CircleSingleLinkedList{
     // 创建一个first节点当前没有编号
     private Boy first = null;
     //添加小孩节点,构建成一个环形的链表
     public void addBoy(int nums) {
 // nums做一个数据校验
         if (nums < 1) {
             System.out.println("nums的值不正确");
             return;
         }
             Boy curBoy = null; //辅助指针,帮助构建环形链表
 //使用for来创建我们的环形链表
             for (int i = 1; i <= nums; i++) {
 // 根据编号,创建小孩节点
                 Boy boy = new Boy(i);
 // 如果是第一个小孩
                 if (i == 1) {
                     first = boy;
                     first.setNext(first);//构成环
                     curBoy = first; //让curBoy指向第一个小孩
                 } else {
                     curBoy.setNext(boy);//
                     boy.setNext(first);//
                     curBoy = boy;
                 }
             }
         }
    
   //遍历当前的环形链表
     public void showBoy() {
     //判断链表是否为空
      if(first== null) {
       System.out.println("没有任何小孩~");
      return;
     }
     //因为first不能动,因此我们仍然使用一个辅助指针完成遍历
      Boy curBoy = first;
      while (true) {
       System.out.printf("小孩的编号%d \n", curBoy.getNo());
       if (curBoy.getNext()==first) {// 说明已经遍历完毕
        break;
      }
       curBoy = curBoy.getNext(); // curBoy后移
       }
      }
    
     public void countBoy(int startNo, int countNum, int nums) {
      //先对数据进行校验
      if (first== null || startNo< 1 || startNo> nums) {
       System.out. println("参数输入有误,请 重新输入");
       return;
      }
      //创建要给辅助指针,帮助完成小孩出圈
      Boy helper= first;
      //需求创建一个辅助指针(变量) helper ,事先应该指向环形链表的最后这个节点
      while (true) {
       if (helper.getNext()==first) { //说明helper 指向最后小孩节点
        break;
       }
       helper = helper.getNext();
      }
      //小孩报数前,先让first 和helper 移动k- 1次
      for(int j= 0;j < startNo- 1;j++) {
      first = first.getNext();
      helper = helper .getNext();
      }
      //当小孩报数时,让first 和helper 指针同时的移动m - 1次,然后出圈
      //这里是一个循环操作,知道圈中只有一个节点
      while(true) {
      if(helper == first) { //说明圈中只有一个节点
       break;
      }
       //让first 和helper 指针同时的移动countNum- 1
       for(int j = 0;j < countNum- 1;j++) {
        first = first. getNext();
        helper = helper.getNext();
       }
       //这时first 指向的节点,就是要出圈的小孩节点
       System.out.printf("小孩%d出圈\n", first.getNo());
       //这时将first指向的小孩节点出圈
       first = first.getNext();
       helper .setNext(first); //
       }
       System.out. printf("最后留在圈中的小孩编号%d \n", first. getNo());
       }
      }
  class Boy {
     //创建一个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;
     }
  }
 问题描述:
在addboy()方法中
 
 

关于单向循环链表的约瑟夫问题(Java实现)的更多相关文章

  1. Java 用单向循环链表实现 约瑟夫问题

    public class lianbiao2 { class Node{ Node next; int number; public Node getNext() { return next; } p ...

  2. C语言单向循环链表解决约瑟夫问题

    据说著名犹太历史学家 Josephus有过以下的故事:在罗马人占领乔塔帕特后,39 个犹太人与Josephus及他的朋友躲到一个洞中,39个犹太人决定宁愿死也不要被敌人抓到,于是决定了一个自杀方式,4 ...

  3. (java实现)单向循环链表

    什么是单向循环链表 单向循环链表基本与单向链表相同,唯一的区别就是单向循环链表的尾节点指向的不是null,而是头节点(注意:不是头指针). 因此,单向循环链表的任何节点的下一部分都不存在NULL值. ...

  4. 使用java的循环单向链表解决约瑟夫问题

    什么是约瑟夫问题 据说著名犹太历史学家 Josephus有过以下的故事:在罗马人占领乔塔帕特后,39 个犹太人与Josephus及他的朋友躲到一个洞中,39个犹太人决定宁愿死也不要被敌人抓到,于是决定 ...

  5. JS数据结构第三篇---双向链表和循环链表之约瑟夫问题

    一.双向链表 在上文<JS数据结构第二篇---链表>中描述的是单向链表.单向链表是指每个节点都存有指向下一个节点的地址,双向链表则是在单向链表的基础上,给每个节点增加一个指向上一个节点的地 ...

  6. PTA 6-15 用单向循环链表实现猴子选大王 (20 分)

    一群猴子要选新猴王.新猴王的选择方法是:让n只候选猴子围成一圈,从某位置起顺序编号为1~n号.每只猴子预先设定一个数(或称定数),用最后一只猴子的定数d,从第一只猴子开始报数,报到d的猴子即退出圈子: ...

  7. Have Fun with Numbers及循环链表(约瑟夫问题)

    1. 循环链表(约瑟夫问题) https://github.com/BodhiXing/Data_Structure 2. Have Fun with Numbers https://pta.pate ...

  8. 基于visual Studio2013解决算法导论之021单向循环链表

     题目 单向循环链表的操作 解决代码及点评 #include <stdio.h> #include <stdlib.h> #include <time.h> ...

  9. Python 单向循环链表

    操作 is_empty() 判断链表是否为空 length() 返回链表的长度 travel() 遍历 add(item) 在头部添加一个节点 append(item) 在尾部添加一个节点 inser ...

随机推荐

  1. 实验一:Linux系统与应用准备

    项目 内容 这个作业属于哪个课程 班级课程 这个作业的要求在哪里 作业要求 学号-学号 17043133-木腾飞 作业学习目标 (1)学习博客园软件开发者学习社区使用技巧和经验:(2)学习Markdo ...

  2. python - 怎样使用 requests 模块发送http请求

    最近在学python自动化,怎样用python发起一个http请求呢? 通过了解 request 模块可以帮助我们发起http请求 步骤: 1.首先import 下 request 模块 2.然后看请 ...

  3. indetityserver4-implicit-grant-types-请求流程叙述-下篇

    上一篇将请求流程描述一遍,这篇将描述一下相关的源码. 1 访问客户端受保护的资源 GET /Home/Secure HTTP/1.1HTTP/1.1 302 Found Date: Tue, 23 O ...

  4. BUUCTF Crypto

    BUUCTF 几道crypto WP [AFCTF2018]Morse 简单的莫尔斯密码,最直观的莫尔斯密码是直接采用空格分割的点和划线,这题稍微绕了一下使用的是斜杠来划分 所以首先将斜杠全部替换为空 ...

  5. C# 根据BackgroundWoker异步模型和ProgressBar控件,自定义进度条控件

    前言 程序开发过程中,难免会有的业务逻辑,或者算法之类产生让人能够感知的耗时操作,例如循环中对复杂逻辑处理;获取数据库百万乃至千万级数据;http请求的时候等...... 用户在使用UI操作并不知道程 ...

  6. SRAM电路工作原理

    近年来,片上存储器发展迅速,根据国际半导体技术路线图(ITRS),随着超深亚微米制造工艺的成熟和纳米工艺的发展,晶体管特征尺寸进一步缩小,半导体存储器在片上存储器上所占的面积比例也越来越高.接下来宇芯 ...

  7. Rocket - subsystem - CrossingWrapper

    https://mp.weixin.qq.com/s/3-MfNJDCIgOBqUbf4fuerQ 简单介绍CrossingWrapper的实现. 1. CrossesToOnlyOneClockDo ...

  8. Splay代码简化版

    皆さん.こんにちは.上一篇文章,我们讲了Splay如何实现.这一篇我们来让我们的伸展树短一点. 上一篇Splay讲解的链接:リンク. 首先还是变量的定义,在这里呢,我把一些小函数也用Define来实现 ...

  9. .NetCore3.1中的WebApi如何配置跨域

    写法 一: 1. 打开Startup.cs,定义静态变量Any,用以配置跨域. private readonly string Any = "Any"; 2. 在Configure ...

  10. Java实现 LeetCode 828 统计子串中的唯一字符(暴力+转数组)

    828. 统计子串中的唯一字符 我们定义了一个函数 countUniqueChars(s) 来统计字符串 s 中的唯一字符,并返回唯一字符的个数. 例如:s = "LEETCODE" ...