【数据结构(三)】单向环形链表和约瑟夫问题(3)

文章目录

  • 1. 单向环形链表应用场景
  • 2. 思路分析
  • 3. 代码实现
    • 3.1. 实现单向环形链表
    • 3.2. 产生出队编号序列
      • 3.2.1. 思路分析
      • 3.2.2. 代码实现


1. 单向环形链表应用场景

Josephu(约瑟夫、约瑟夫环) 问题:

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

在这里插入图片描述

丢手帕问题
    

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

假设:n=5,即有5人;k=1,从第一个人开始报数;m=2,数到2出列。

单向环形链表完成约瑟夫问题
在这里插入图片描述

出圈顺序为:2–>4–>1–>5–>3


2. 思路分析

构建一个单向环形链表的思路:

①先创建第一个节点(如下图的节点1), 让 first 指向该节点,并形成环形

在这里插入图片描述

②后面当我们每创建一个新的节点(如下图的节点2),就把该节点加入到已有的环形链表中即可

在这里插入图片描述
③重复步骤2(假设重复两次,形成如下4个节点的单向环形链表)

在这里插入图片描述

遍历环形链表

  1. 先让一个**辅助指针(**变量) curBoy,指向first节点
  2. 然后通过一个while循环遍历 该环形链表即可 curBoy.next == first 结束

3. 代码实现

3.1. 实现单向环形链表

package Linkedlist;public class Josepfu {public static void main(String[] args) {CircleSingleLinkedList circleSingleLinkedList = new CircleSingleLinkedList();circleSingleLinkedList.addBoy(5);//加入5个小孩节点circleSingleLinkedList.showBoy();}}//创建一个环形的单向链表
class CircleSingleLinkedList {//创建一个first节点,当前没有编号private Boy first = new Boy(-1);//添加小孩,构建成一个环形的链表public void addBoy(int nums) {//对 nums 做一个数据校验(不能小于1)if (nums < 1) {System.out.println("nums的值不正确,不能小于1");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后移}}
}//创建一个Boy类,表示一个节点
class Boy{private int no;//编号private Boy next;//指向下一个节点,默认nullpublic Boy(int no){this.no = no;}public int getNo() {return this.no;}public void setNo(int no) {this.no = no;}public Boy getNext() {return this.next;}public void setNext(Boy next) {this.next = next;}}

运行结果:

在这里插入图片描述

3.2. 产生出队编号序列

3.2.1. 思路分析

①需求创建一个辅助指针(变量) helper , 事先应该指向环形链表的最后这个节点(小孩报数前,先让 first 和 helper 移动 k - 1次)

在这里插入图片描述

②当小孩报数时,让first 和 helper 指针同时 的移动 m - 1 次

(本题m=2,故移动m-1=1次)

在这里插入图片描述

③这时就可以将first指向的小孩节点 出圈
first = first .next
helper.next = first
(原来first 指向的节点就没有任何引用,就会被回收)

在这里插入图片描述

出圈的顺序: 2->4->1->5->3

3.2.2. 代码实现

package Linkedlist;public class Josepfu {public static void main(String[] args) {CircleSingleLinkedList circleSingleLinkedList = new CircleSingleLinkedList();circleSingleLinkedList.addBoy(5);// 加入5个小孩节点circleSingleLinkedList.showBoy();// 出圈circleSingleLinkedList.countBoy(1, 2, 5);// 2->4->1->5->3}}// 创建一个环形的单向链表
class CircleSingleLinkedList {// 创建一个first节点,当前没有编号private Boy first = new Boy(-1);// 添加小孩,构建成一个环形的链表public void addBoy(int nums) {// 对 nums 做一个数据校验(不能小于1)if (nums < 1) {System.out.println("nums的值不正确,不能小于1");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后移}}// 根据输入,计算出小孩出圈的顺序/*** * @param startNo  表示从第几个小孩开始数数* @param countNum 表示数几下* @param nums     表示多少个小孩在圈中*/public void countBoy(int startNo, int countNum, int nums) {// 先对数据进行校验if (first == null || startNo < 1 || startNo > nums) {System.out.println("参数输入有误,请重新输入");return;}// 创建一个辅助指针 helper 帮助完成小孩出圈Boy helper = first;// 创建一个辅助指针(变量)helper,事先应该指向环形链表的最后这个节点while (true) {if (helper.getNext() == first) {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-1for (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());}}// 创建一个Boy类,表示一个节点
class Boy {private int no;// 编号private Boy next;// 指向下一个节点,默认nullpublic Boy(int no) {this.no = no;}public int getNo() {return this.no;}public void setNo(int no) {this.no = no;}public Boy getNext() {return this.next;}public void setNext(Boy next) {this.next = next;}}

运行结果:

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

mac 和 windows 相互传输文件【共享文件夹】

文章目录 前言创建共享文件夹mac 连接共享文件夹 前言 温馨提示&#xff1a;mac 电脑和 windows 电脑必须处于同一局域网下 本文根据创建共享文件夹的方式实现文件互相传输&#xff0c;所以两台电脑必须处于同一网络 windows 创建共享文件夹&#xff0c;mac 电脑通过 windows…

六、程序员指南:数据平面开发套件

PORT HOTPLUG FRAMEWORK 端口热插拔框架为DPDK应用程序提供在运行时附加和分离端口的能力。由于该框架依赖于PMD实现&#xff0c;PMD无法处理的端口超出了该框架的范围。此外&#xff0c;在从DPDK应用程序分离端口后&#xff0c;该框架不提供从系统中移除设备的方法。对于由物…

【DevOps】Git 图文详解(七):标签管理

Git 图文详解&#xff08;七&#xff09;&#xff1a;标签管理 标签&#xff08;Tags&#xff09;指的是某个分支某个特定时间点的状态&#xff0c;是对某一个提交记录的 固定 “指针” 引用。一经创建&#xff0c;不可移动&#xff0c;存储在工作区根目录下 .git\refs\tags。可…

【vue+eltable】修改表格滚动条样式

<style lang"scss" scoped> ::v-deep .el-table__body-wrapper::-webkit-scrollbar {width: 10px; /*纵向滚动条的宽度*/height: 10px; /*横向滚动条的高度*/ } /*定义滚动条轨道 内阴影圆角*/ ::v-deep .el-table__body-wrapper::-webkit-scrollbar-track {bo…

开源之夏2023 MatrixOne 项目结业啦

开源之夏是由中国科学院软件研究所与 OpenEuler 社区共同主办的一项面向高校学生的暑期在线活动&#xff0c;旨在鼓励在校学生积极参与开源软件的开发维护&#xff0c;促进优秀开源软件社区的蓬勃发展。 在开源之夏 2023 年中&#xff0c;MatrixOne 一共有 2 个任务项目&#…

Python 和 Ruby 谁是最好的Web开发语言?

Python 和 Ruby 都是目前用来开发 websites、web-based apps 和 web services 的流行编程语言之一。 【这个时候又人要说PHP是世界上最好的语言了】 我就不说PHP 最好的方法 VS 以人为本的语言 社区: 稳定与创新 尽管特性和编程哲学是选择一个语言的首要驱动因素&#xff0c…

ruoyi-vue前后端分离版本验证码实现思路

序 时隔三个月&#xff0c;再次拿起我的键盘。 前言 ruoyi-vue是若依前后端分离版本的快速开发框架&#xff0c;适合用于项目开始搭建后台管理系统。本篇文章主要介绍其验证码实现的思路。 一、实现思路简介 1、后端会生成一个表达式&#xff0c;比如1 2 ? 3&#xff0…

AT89S52单片机

目录 一.AT89S52单片机的硬件组成 1.CPU(微处理器) (1)运算器 (2)控制器 2.数据存储器 (RAM) (1)片内数据存储器 (2)片外数据存储器 3.程序存储器(Flash ROM) 4.定时器/计数器 5.中断系统 6.串行口 7.P0口、P1口、P2口和P3口 8.特殊功能寄存器 (SFR) 常用的特殊功…

外部 prometheus监控k8s集群资源

prometheus监控k8s集群资源 一&#xff0c;通过CADvisior 监控pod的资源状态1.1 授权外边用户可以访问prometheus接口。1.2 获取token保存1.3 配置prometheus.yml 启动并查看状态1.4 Grafana 导入仪表盘 二&#xff0c;通过kube-state-metrics 监控k8s资源状态2.1 部署 kube-st…

[架构之路-247]:目标系统 - 设计方法 - 软件工程 - 结构化方法的基本思想、本质、特点以及在软件开发、在生活中的应用

目录 前言&#xff1a; 一、什么是非结构化方法 1.1 什么是非结构化方法 1.2 非结构化方法的适用场合 二、什么是结构化方法 1.1 结构化方法诞生的背景&#xff1a;软件规模发展&#xff1a;大规模、复杂系统的需要 1.2 概述 1.3 主要特点与核心思想 三、结构化方法在…

flutter iOS 视频mov格式转MP4格式

flutter iOS 视频mov格式转MP4格式 前言一、使用video_compress压缩视频总结 前言 今天在写项目的时候&#xff0c;突然发现iOS 里面的有些视频格式是mov的格式&#xff0c;这就导致在视频播放组件无法播放的问题&#xff0c;期间试过替换视频格式&#xff0c;但是又不想存储文…

注解方式优雅的实现 Redisson 分布式锁

1前言 日常开发中&#xff0c;难免遇到一些并发的场景&#xff0c;为了保证接口执行的一致性&#xff0c;通常采用加锁的方式&#xff0c;因为服务是分布式部署模式&#xff0c;本地锁Reentrantlock和Synchnorized这些就先放到一边了&#xff0c;Redis的setnx锁存在无法抱保证…

Ubuntu Server download

前言 Ubuntu——公共云、数据中心和边缘上最受欢迎的 Linux 发行版。自成立以来&#xff0c;Ubuntu 一直在获得市场份额&#xff0c;截至今天已接近 50%。 Ubuntu Server download VersionUbuntu Server 其它主机型号版本Ubuntu AMD历史版下载百度云Ubuntu Server all Ubuntu…

MT8735/MTK8735安卓核心板规格参数介绍

MT8735核心板是一款高性能的64位Cortex-A53四核处理器&#xff0c;设计用于在4G智能设备上运行安卓操作系统。这款多功能核心板支持LTE-FDD/LTE-TDD/WCDMA/TD-SCDMA/EVDO/CDMA/GSM等多种网络标准&#xff0c;同时还具备WiFi 802.11a/b/g/n和BT4.0LE等无线通信功能。此外&#x…

O-Star|再相识

暑去秋来&#xff0c;岁月如梭&#xff0c;几名"O-Star"们已经入职一段时间&#xff0c;在这期间他们褪去青涩&#xff0c;逐渐适应了公司的工作环境和文化&#xff0c;迈向沉稳&#xff5e; 为了进一步加深校招生之间的交流与了解&#xff0c;提高校招生的凝聚力和…

LangChain 6根据图片生成推广文案HuggingFace中的image-caption模型

根据图片生成推广文案&#xff0c; 用的HuggingFace中的image-caption模型 LangChain 实现给动物取名字&#xff0c;LangChain 2模块化prompt template并用streamlit生成网站 实现给动物取名字LangChain 3使用Agent访问Wikipedia和llm-math计算狗的平均年龄LangChain 4用向量数…

Libvirt-Qemu-Kvm 操作手记

(持续更新~) 本文主要用于记录在操作libvirt qemu kvm过程中遇到的问题及原因分析。 Hugepage 让qemu使用大页可以减少tdp的size&#xff0c;一定程度上可以提高性能&#xff1b;使用大页可以用memfd或者file backend。 memfd 操作步骤如下&#xff1a; 在系统中reserv…

idea Maven Helper插件使用方法

idea Maven Helper插件使用方法 文章目录 idea Maven Helper插件使用方法&#x1f4c6;1.安装mavenhelper&#x1f5a5;️2.使用教程&#x1f4cc;3.解决冲突&#x1f4c7;4.列表展示依赖&#x1f9e3;5.tree展示依赖&#x1f5a5;️6.搜索依赖&#x1f58a;️7.最后总结 &…

Springboot+vue的社区医院管理系统(有报告),Javaee项目,springboot vue前后端分离项目

演示视频&#xff1a; Springbootvue的社区医院管理系统(有报告)&#xff0c;Javaee项目&#xff0c;springboot vue前后端分离项目 项目介绍&#xff1a; 本文设计了一个基于Springbootvue的前后端分离的应急物资管理系统&#xff0c;采用M&#xff08;model&#xff09;V&am…

Logstash同步MySQL数据到ES

简介 1.1 什么是Logstash&#xff1f; Logstash作为一个具备实时流水线功能的开源数据收集引擎&#xff0c;拥有强大的能力。它能够从不同来源收集数据&#xff0c;并将其动态地汇聚&#xff0c;进而根据我们定义的规范进行转换或者输出到我们定义的目标地址。 1.2 Logstash的…