约瑟夫问题又名丢手绢问题。相传著名犹太历史学家 Josephus 利用其规则躲过了一场自杀游戏,而后投降了罗马。

问题:

已知n个人(以编号1,2,3...n分别表示)围坐在一张圆桌周围。
* 从编号为k的人开始报数,数到m的那个人出列;
* 他的下一个人又从1开始报数,数到m的那个人又出列;
* 依此规律重复下去,直到圆桌周围的人全部出列。 用节点来模拟游戏中的人,用链表来表示游戏中的人按一定的顺序排列。每一个节点给一个编号,从编号为k的节点开始计数,计到m的节点从链表中退出。
之后m的下一个节点从新开始计数,依次...
计数为m的节点要被从链表中删除,只需要将m的上一个节点的引用指向m的下一个节点便可。
依上可知我们的节点类只需要两个属性,一个表示它自己的编号,另一个是引用,指向下一个节点,最后的节点指向第一个节点。
对于第一个节点必须要有一个入口节点————否则无法进入环形链表。

我们需要一个跑龙套的节点以便处理游戏中的计数。
由于是单向链表,所以当我们找到计数为m的节点时还要定位m的上一个节点,并更新其引用,使得计数为m的节点失去引用后被java垃圾回收机制处理掉。
要定义m的上一个节点,可以依链表方向,重新遍历一遍到一个引用指向计数为m的节点,该节点便是。
          还可以将链表换成双向链表,也可以定义两个跑龙套的节点一个用以计数,一个随后以便定位m的上一个节点。
这些方法这里都不用,由于找到计数为m的节点并不需要对它做任何处理,而是处理它的上一个节点————将其引用更新为m的下一个节点,
所以我们可以简而计数到m-1便可直接处理。
给出完整代码如下:
public class JosephCycle {
public static void main(String []args){
CycleLink cl = new CycleLink();
cl.setLen(7);
cl.setK(3);
cl.setM(3);
cl.creatLink();
cl.play();
cl.show();
} } class Node{
int num;
Node nextNode = null;
public Node(int num){
this.num = num;
}
} class CycleLink{
int len;
int k ;
int m ;
Node firstNode = null;
Node temp = null;
public void setLen(int len){
this.len = len;
} public void setK(int k){
this.k = k;
}
public void setM(int m){
this.m =m;
} //创建链表
public void creatLink(){
for(int i = 1; i <= len ; i++){
// 处理首节点
if(i==1){
Node nd= new Node(i);
firstNode = nd;
temp = nd;
}else if(i == len){ //处理末节点
Node nd = new Node(i);
temp.nextNode = nd;
temp = nd;
temp.nextNode = firstNode;
}else{
Node nd = new Node(i);
temp.nextNode = nd;
temp = nd;
}
}
} public void play(){
temp = firstNode;
// 先找到编号为k的节点
for(int i = 1 ; i < k; i++){
temp = temp.nextNode;
}
while(this.len !=1){
//报数,找到m的上一个节点
for(int j = 1 ;j < m-1; j++){
temp = temp.nextNode;
} //因为少报了 1 ,所以将下一个节点删除,并从下下一个节点重新开始报数
System.out.println("要删除的节点: "+temp.nextNode.num);
/**
* 如果删除节点是firstNode,则将firstNode更新为下一个节点
* 注意不能用编号判断,因为新的编号对应的节点有可能又被删除
*/
if(temp.nextNode==firstNode){
firstNode = temp.nextNode.nextNode;
}
temp.nextNode = temp.nextNode.nextNode;
temp = temp.nextNode;
//this.show();
this.len--;
} } /**
* 遍历链表打印整个链表
*/
public void show(){
temp = firstNode;
do{
System.out.println(temp.num);
temp = temp.nextNode;
}while(temp != firstNode);
}
}

这段代码可拿来直接运行,运行结果:

    

最后剩余的节点为:6

在实际应用中,比如排队,根据不同的k,m,len的值,对于有些特殊情况,可能有不同的算法我也不知道,猜测如此!

为何有此猜想,读者可参看:约瑟夫环问题详解

 

java 环形链表实现约瑟夫(Joseph)问题的更多相关文章

  1. PHP算法学习(8) 环形链表 解决约瑟夫问题

    2019年2月25日17:29:17 Josephus有过的故事:39 个犹太人与Josephus及他的朋友躲到一个洞中,39个犹太人决定宁愿死也不要被敌人抓.于是决定了自杀方式,41个人排成一个圆圈 ...

  2. php实现单,双向链表,环形链表解决约瑟夫问题

    传智播客PHP学院 韩顺平 PHP程序员玩转算法第一季  http://php.itcast.cn 聊天篇: 数学对我们编程来说,重不重要? 看你站在什么样的层次来说. 如果你应用程序开发,对数学要求 ...

  3. 单向环形链表解决约瑟夫环(Josephus)问题

    一.约瑟夫环问题 Josephu 问题为:设编号为1,2,- n的n个人围坐一圈,约定编号为k(1<=k<=n)的人从1开始报数,数到m 的那个人出列,它的下一位又从1开始报数,数到m的那 ...

  4. 数据结构与算法——链表 Linked List(单链表、双向链表、单向环形链表-Josephu 问题)

    链表是有序的列表,但是在内存中存储图下图所示 链表是以 节点 的方式来存储,是 链式存储 每个节点包含 data 域.next 域,指向下一个节点 链表的各个节点 不一定是连续存储,如上图所示 链表还 ...

  5. Java数据结构之单向环形链表(解决Josephu约瑟夫环问题)

    1.Josephu(约瑟夫.约瑟夫环)问题: 设编号为1,2,… n的n个人围坐一圈,约定编号为k(1<=k<=n)的人从1开始报数,数到m 的那个人出列,它的下一位又从1开始报数,数到m ...

  6. 3,java数据结构和算法:约瑟夫环出队顺序, 单向环形链表的应用

    什么是约瑟夫环? 就是数小孩游戏: 直接上代码: 要实现这个,只需要理清思路就好了 孩子节点: class Boy{ int no;//当前孩子的编码 Boy next; // 下一节点 public ...

  7. 【java基础 12】HashMap中是如何形成环形链表的?

    导读:经过前面的博客总结,可以知道的是,HashMap是有一个一维数组和一个链表组成,从而得知,在解决冲突问题时,hashmap选择的是链地址法.为什么HashMap会用一个数组这链表组成,当时给出的 ...

  8. 图解Java数据结构之环形链表

    本篇文章介绍数据结构中的环形链表. 介绍 环形链表,类似于单链表,也是一种链式存储结构,环形链表由单链表演化过来.单链表的最后一个结点的链域指向NULL,而环形链表的建立,不要专门的头结点,让最后一个 ...

  9. 丢手帕问题(环形链表)---Java 待优化

    /** * * @author Administrator * 功能:丢手帕问题 */ package com.litao; public class Demo4 { /** * @param arg ...

随机推荐

  1. Docker Weave 介绍 or 工作原理

    Docker Weave Network Weave Network:属于第三方网络项目. Weave在Docker主机之间实现Overlay网络,使用业界标准VXLAN封装,基于UDP传输,也可以加 ...

  2. 清晰的教你如何将 Maven 项目上传至 中央仓库以及版本更新

    项目中总会依赖一些大牛的开源库,例如 Android 开发 通过以下方式就可以方便的引入库.那么如何将我们自己的库也能这样来使用呢?下面我们将详细的讲解每一步的过程: implementation ' ...

  3. 服务列表中找不到mysql

    服务列表中找不到mysql - 解决办法 1.在开始处输入cmd,找到cmd选择以管理员身份运行(必须以管理员运行,直接win+r打开无效) 2.进入到MySQL安装目录的bin目录 3.执行mysq ...

  4. backbone点滴

    可以查看http://www.css88.com/doc/backbone/ backbone与angular http://www.infoq.com/cn/articles/backbone-vs ...

  5. 第 8 章 容器网络 - 065 - 容器在 Weave 中如何通信和隔离?

    首先在host2 执行如下命令: weave launch 10.12.31.22 这里必须指定 host1 的 IP 10.12.31.22,这样 host1 和 host2 才能加入到同一个 we ...

  6. obs studio 使用

    专业,开源,无广告,免费,录屏/推流神器--obs studio 稍微简单的也有captura, 原理:调用本地API获取音频流,图像流(全屏幕,单个windows窗口的图像输出)->开源音视频 ...

  7. eclipse 运行 mapreduce程序报错 No job jar file set. User classes may not be found. See JobConf(Class) or JobConf#setJar(String).

    报错信息 17/07/06 17:00:27 WARN mapred.JobClient: Use GenericOptionsParser for parsing the arguments. Ap ...

  8. MySql 8.0 版本使用navicat连不上解决

    先通过命令行进入mysql的root账户: 更改加密方式 ALTER USER 'root'@'localhost' IDENTIFIED BY 'password' PASSWORD EXPIRE ...

  9. css新单位vw,vh在响应式设计中的应用

    考虑到未来响应式设计的开发,如果你需要,浏览器的高度也可以基于百分比值调整.但使用基于百分比值并不总是相对于浏览器窗口的大小定义的最佳方式,比如字体大小不会随着你窗口改变而改变,如今css3引入的新单 ...

  10. Android测试(三)——APK文件反编译

    APK文件反编译: 在进行反编译操作前,先简单介绍下smali文件: smali是一种文件格式,语法和Jasmine的语言类似,这些smali文件包含开发应用程序时编写的java类的代码. 工具:ja ...