约瑟夫问题的环形链表实现[Java]

⭐作者介绍:大二本科网络工程专业在读,持续学习Java,努力输出优质文章
⭐作者主页:@逐梦苍穹
⭐如果觉得文章写的不错,欢迎点个关注一键三连😉有写的不好的地方也欢迎指正,一同进步😁

约瑟夫问题

  • 1、背景介绍
  • 2、问题
  • 3、实现过程
  • 4、全部代码

1、背景介绍

  约瑟夫问题(Josephus Problem)是一个古老而有趣的数学问题,源自于犹太历史学家弗拉维奥·约瑟夫斯(Flavius Josephus)。问题的描述如下:
  在古代,约瑟夫斯是一个被罗马占领的犹太人城市的叛乱者。为了避免被捕,他与一群同伴藏身于一个山洞中。但是,当他们被罗马军队发现并围困在山洞里时,他们必须决定是被俘还是选择集体自杀。
  为了做出决定,约瑟夫斯提出了一种方法来确定哪些人将被选中。他让所有人站成一个圆圈,并从一个人开始,按照一定的规则逐个数数。每数到第 m 个人时,该人将被杀掉,然后从下一个人开始重新数数。这个过程一直进行下去,直到只剩下最后一个人。
  约瑟夫问题的关键是确定起始位置和数数的规则。通常起始位置用 K 表示,数数规则用 M 表示,即每数到第 M 个人时,该人被杀掉。问题的目标是确定最后剩下的人的位置。
  约瑟夫问题可以用数学公式来解决,也可以通过模拟环形链表的方式来求解。无论使用哪种方法,解决约瑟夫问题都需要一定的算法思想和数学推导。
  约瑟夫问题的应用非常广泛,涉及到计算机科学、数学、密码学等领域。解决约瑟夫问题的算法思想也可以应用于任务调度、资源分配等实际问题中。

2、问题

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

3、实现过程

在环形链表的基础上进行修改,得到JosephusCircularLinkedList。改动地方如下:
在这里插入图片描述

同时,增加了一个方法countOrder。这是实现约瑟夫问题的主要方法,详细注释版本如下:
在这里插入图片描述

对该方法的解析:
  该方法根据用户输入的参数,计算出小孩出圈的顺序。首先进行参数校验,确保输入的参数合法。然后创建一个辅助指针helper,用于辅助完成小孩出圈的操作。通过遍历找到环形链表的最后一个节点,并将helper指向最后一个小孩节点。
  接下来,根据起始位置start,将firstNode和helper指针分别移动(start - 1)次,使它们指向起始位置的小孩节点。
  然后,通过循环操作,每次让firstNode和helper指针同时移动(countNum - 1)次,表示小孩报数。当helper和firstNode指向同一个节点时,说明圈中只剩下一个节点,循环结束。
  在循环过程中,每次移动完成后,firstNode指向的节点即为要出圈的小孩节点,输出其编号。
  最后,将firstNode指向的小孩节点出圈,即将其从链表中移除,并更新helper的指向。循环结束后,输出最后留在圈中的小孩的编号。

这段代码实现了约瑟夫问题的解决方案,根据起始位置和报数规则,依次输出出圈的小孩编号,直到最后只剩下一个小孩。

4、全部代码

package linkedList.circularLinkedList.Josephu;

import linkedList.circularLinkedList.Node;

/**
 * @author 逐梦苍穹
 * @date 2023/5/22 14:48
 */
public class JosephusCircularLinkedList {
    private Node firstNode = null;

    /**
     * 添加
     */
    public void add(int studentsNums) {
        // nums 做一个数据校验
        if (studentsNums < 1) {
            System.out.println("nums的值不正确");
            return;
        }
        Node temp = null; // 辅助指针,帮助构建环形链表
        // 使用for来创建我们的环形链表
        for (int i = 1; i <= studentsNums; i++) {
            // 根据编号,创建小孩节点
            Node node = new Node(i);
            // 如果是第一个小孩
            if (i == 1) {
                firstNode = node;
                firstNode.setNext(firstNode); // 构成环
                temp = firstNode; // temp指向第一个小孩
            } else {
                temp.setNext(node);
                node.setNext(firstNode);
                temp = node;
            }
        }
    }

    /**
     * 根据用户的输入,计算出小孩出圈的顺序
     * @param start 表示从第几个小孩开始数数(K)
     * @param countNum 表示数几下(m)
     * @param InitialNums 表示最初有多少小孩在圈中
     */
    public void countOrder(int start, int countNum, int InitialNums) {
        // 先对数据进行校验
        if (firstNode == null || start < 1 || start > InitialNums) {
            System.out.println("参数输入有误, 请重新输入");
            return;
        }
        // 创建要给辅助指针,帮助完成小孩出圈
        Node helper = firstNode;
        // 需求创建一个辅助指针(变量) helper , 事先应该指向环形链表的最后这个节点
        while (true) {
            if (helper.getNext() == firstNode) { // 说明helper指向最后小孩节点
                break;
            }
            helper = helper.getNext();
        }
        //小孩报数前,先让 first 和  helper 移动 k - 1次
        for(int j = 0; j < start - 1; j++) {
            firstNode = firstNode.getNext();
            helper = helper.getNext();
        }
        //当小孩报数时,让first 和 helper 指针同时 的移动  m  - 1 次, 然后出圈
        //这里是一个循环操作,知道圈中只有一个节点
        while(true) {
            if(helper == firstNode) { //说明圈中只有一个节点
                break;
            }
            //让 first 和 helper 指针同时 的移动 countNum - 1
            for(int j = 0; j < countNum - 1; j++) {
                firstNode = firstNode.getNext();
                helper = helper.getNext();
            }
            //这时first指向的节点,就是要出圈的小孩节点
            System.out.printf("小孩%d出圈\n", firstNode.getId());
            //这时将first指向的小孩节点出圈
            firstNode = firstNode.getNext();
            helper.setNext(firstNode); //

        }
        System.out.printf("最后留在圈中的小孩编号%d \n", firstNode.getId());

    }

    /**
     * 删除
     */
    public void delete(int id) {
        Node temp = this.firstNode;
        if (temp == null) {
            System.out.println("链表空无法删除");
            return;
        }
        while (true) {
            if (temp.getNext() == firstNode) break;
            if (temp.getNext().getId() == id) {
                temp.setNext(temp.getNext().getNext());
                System.out.println("Node{id=" + id + "}:删除成功");
                return;
            }
            temp = temp.getNext();
        }
        System.out.println("无此节点");
    }

    /**
     * 显示
     */
    public void show(){
        if (firstNode == null) {
            System.out.println("环形链表为空无法显示");
            return;
        }
        // 因为firstNode不能动,因此我们仍然使用一个辅助指针完成遍历
        Node temp = firstNode;
        while (true) {
            System.out.println(temp);
            if (temp.getNext() == firstNode) {// 说明已经遍历完毕
                break;
            }
            temp = temp.getNext(); // temp后移
        }
    }
}

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/22926.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

C++入门预备语法

C入门预备语法 C关键字命名空间C输入&输出初步缺省参数函数重载引用内联函数auto和范围for&#xff08;C11&#xff09;指针空值nullptr C关键字 命名空间 命名空间是一种将变量名、函数名、类名和库名称等封装到一个命名空间域中&#xff0c;与其他域的同名量相隔离&…

Go语言环境搭建(内附网P下载地址)

一、Golang语言的官网 首先我们登录Golang的官方网站&#xff1a;https://golang.org/ 因为Google和中国的关系&#xff0c;直接登录Golang的官网&#xff0c;需要翻墙。 当然你也可以登录Golang的国内网站&#xff1a;https://golang.google.cn/ 二、下载 在Mac、Windows和L…

如何基于LiveNVR实现无人机等RTMP推流转成GB28181协议级联到GB28181视频平台

1、需求介绍 目前很多移动终端设备&#xff08;如无人机等&#xff09;只支持RTMP推流输出&#xff0c;不支持GB28181协议。但是又有需要通过GB28181协议接入到视频平台的需求。比如有些大疆无人机产品不能直接注册国标平台&#xff0c;只能rtmp推流。那么&#xff0c;项目中如…

Linux-搭建web服务器

综合练习&#xff1a;请给openlab搭建web网站 ​ 网站需求&#xff1a; ​ 1.基于域名[www.openlab.com](http://www.openlab.com)可以访问网站内容为 welcome to openlab!!! ​ 2.给该公司创建三个子界面分别显示学生信息&#xff0c;教学资料和缴费网站&#xff0c;基于[www.…

javaWebssh车辆保养管理系统myeclipse开发mysql数据库MVC模式java编程计算机网页设计

一、源码特点 java ssh车辆保养管理系统是一套完善的web设计系统&#xff08;系统采用ssh框架进行设计开发&#xff09;&#xff0c;对理解JSP java编程开发语言有帮助&#xff0c;系统具有完整的源代码和数据库&#xff0c;系统主要采用 B/S模式开发。开发环境为TOMCAT7.…

湍流的数值模拟方法概述

湍流&#xff0c;又称紊流&#xff0c;是一种极其复杂、极不规则、极不稳定的三维流动。湍流场内充满着尺度大小不同的旋涡&#xff0c;大旋涡尺度可以与整个流畅区域相当&#xff0c;而小漩涡尺度往往只有流场尺度千分之一的数量级&#xff0c;最小尺度旋涡的尺度通过其耗散掉…

IO流详解

IO流 1. 文件 1.1 什么是文件 文件对大家来说都不陌生&#xff1a; 文件是保存数据的地方&#xff0c;它可以保存文字、图片、视频等等例如大家平时使用的word文档、Excel文档、PPT文档等都是文件 1.2 文件流 文件在程序中是以流的形式来操作的流是指数据在数据源&#x…

高通开发系列 - 音频驱动中的APR通道不能打开问题

By: fulinux E-mail: fulinux@sina.com Blog: https://blog.csdn.net/fulinus 喜欢的盆友欢迎点赞和订阅! 你的喜欢就是我写作的动力! 目录 问题概述问题现象问题分析问题解决问题概述 对msm8909平台进行内核升级,相应的其音频驱动也需要进行升级,使用了同平台的音频驱动作…

ZBX_NOTSUPPORTED: Unsupported item key.

问题 ZBX_NOTSUPPORTED: Unsupported item key. 详细问题 笔者安装zabbix后&#xff0c;自定义item key进行测试。需在zabbix-server 端 切换目录&#xff1a; cd /usr/local/zabbix/bin 执行查询命令&#xff1a; ./zabbix_get -s 192.168.174.136 -p 10050 -k “home.file…

六、数据仓库详细介绍(ETL)经验篇

0x00 前言 日常工作中大多数时候都是在做数据开发&#xff0c;ETL 无处不在。虽然最近两年主要做的大数据开发&#xff0c;但感觉日常干的这些还是 ETL 那点事儿&#xff0c;区别只是技术组件全换了、数据量大了很多。 前几年数仓势微&#xff0c;是因为传统的那些工具数据库等…

LeetCode 117. 填充每个节点的下一个右侧节点指针 II

117. 填充每个节点的下一个右侧节点指针 II 描述 给定一个二叉树&#xff1a; struct Node {int val;Node *left;Node *right;Node *next; }填充它的每个 next 指针&#xff0c;让这个指针指向其下一个右侧节点。如果找不到下一个右侧节点&#xff0c;则将 next 指针设置为 …

opencv缺陷检测

随着自动化生产设备的普及&#xff0c;工业机器人在各行各业的应用也越来越广泛&#xff0c;越来越多的生产线由自动化设备取代人工操作&#xff0c;实现自动化生产。在机器人分拣过程中&#xff0c;机器人不仅可以将不同规格和质量的产品准确地放入指定的托盘中&#xff0c;而…

Puppeteer入门实践

环境 1、安装nodejs 官网&#xff1a;https://nodejs.org/zh-cn 下载安装好nodejs只后 验证&#xff1a;node -v 出现版本号表示安装成功&#xff0c;否则需要配置环境变量 2、创建node项目并初始化 随便新建一个文件夹 进入文件夹搜索cmd回车 执行npm init -y 安装依赖 …

软件测试基础知识整理(八)- 软件缺陷

目录 一、软件缺陷 1.1 缺陷定义 1.2 缺陷判定标准 1.3 软件缺陷产生的原因 1.4 软件缺陷产生的根源 1.5 软件缺陷信息 1.5.1 缺陷状态 1.5.2 缺陷严重程度 1.5.3 缺陷优先级 1.6 缺陷报告模板 1.7 缺陷报告注意事项 1.8 缺陷跟踪流程 1.9 缺陷数据分析关注的问题 …

【ETH】以太网----PHY芯片LAN8720A----电路原理图

一、LAN8720A----简介 LAN8720A 是低功耗的 10/100M 以太网 PHY 层芯片&#xff0c;I/0 引脚电压符合EEE802.3-2005 标准&#xff0c;支持通过 RMI 接口与以太网 MAC 层通信&#xff0c;内置 10-BASE-T/100BASE-TX 全双工传输模块&#xff0c;支持 10Mbps 和 100Mbps。 LAN87…

内蒙古自治区住房和城乡建设分析及解决方案

安科瑞 徐浩竣 江苏安科瑞电器制造有限公司 zx acrelxhj 摘 要&#xff1a;为深入贯彻落实《国务院办公厅关于印发新能源汽车产业发展规划&#xff08;2021—2035年&#xff09;的通知》&#xff08;国办发 ﹝2020﹞39号&#xff09;、《国家发展改革委等部门关于进一步提升…

java前后端分离有详细内容吗?

微服务架构java前后端分离都有哪些具体内容&#xff1f;目前&#xff0c;有不少客户朋友经常询问我们类似的问题。其实&#xff0c;在新的经济发展形势下&#xff0c;提质增效的低代码开发平台微服务架构早已成为不少新老客户的选择&#xff0c;它们不仅能提高办公协作效率&…

多商户商城系统开发功能优势与选择技巧

电商行业的持续发展&#xff0c;让越来越多的商家企业开始选择入驻多商户商城&#xff0c;通过该系统不仅能够为消费者提供更加便捷良好的购物体验&#xff0c;而且也能够为企业提供一个高效稳定的电商平台&#xff0c;可以说是未来电商行业发展的重要趋势。那么多商户商城系统…

Spring之DI(依赖注入)

依赖注入&#xff08;DI&#xff09;是一个过程&#xff0c;在这个过程中&#xff0c;对象仅通过构造函数参数、工厂方法的参数或在对象被实例化后通过属性设置来定义它们的依赖项&#xff08;即与该对象一起工作的其他对象&#xff09;。然后&#xff0c;容器在创建 bean 时注…

Hadoop HA(高可用)搭建

ZooKeeper配置 解压安装 添加ZK环境变量 分发文件 启动 安装配置 Hadoop 解压安装 修改hadoop-env.sh文件 修改Hadoop配置文件core-site.xml HDFS 配置文件hdfs-site.xml MapReduce 配置文件 mapred-site.xml YARN 配置文件yarn-site.xml 配置worekers 分发配…
最新文章