Java用单向环形链表来解决约瑟夫环Josepfu问题

 更新时间:2021年10月19日 08:36:44   作者:叶绿体不忘呼吸  
如果把单链表的最后一个节点的指针指向链表头部,而不是指向NULL,那么就构成了一个单向循环链表,通俗讲就是把尾节点的下一跳指向头结点

简单介绍

如果把单链表的最后一个节点的指针指向链表头部,而不是指向NULL,那么就构成了一个单向循环链表,通俗讲就是让尾节点指向头结点。

在这里插入图片描述

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

代码实现

节点类

//节点类
class JNode {
    private int id;
    private JNode next;

    public JNode(int id) {
        this.id = id;
    }

    public int getId() {
        return id;
    }

    public JNode getNext() {
        return next;
    }

    public void setNext(JNode next) {
        this.next = next;
    }
    
}

链表类(包括节点管理和约瑟夫环问题解决)

//链表类
class CircleSingleLinkedList {
    private JNode first = null; //定义第一个节点,未创建时为null

    //添加节点,构建环形链表
    public void add(int num) {
        if (num < 1){
            System.out.println("创建个数不符合规定!");
            return;
        }
        JNode curNode = null; //辅助变量
        for (int i = 1; i <= num; i++) {
            JNode newNode = new JNode(i);
            if (i == 1){ //第一个节点较为特殊
                first = newNode; //真正创建了第一个节点
                first.setNext(first); //形成环状
                curNode = first; //让辅助变量开始作用
            }else { //第二个及其之后节点
                curNode.setNext(newNode); //让当前节点指向新建的节点
                newNode.setNext(first); //让新建的节点指向第一个节点,形成环状
                curNode = newNode; //更新辅助变量
            }
        }
    }

    //遍历链表
    public void list(){
        if (first == null){
            System.out.println("链表为空!");
            return;
        }
        JNode temp = first;
        while (true){
            System.out.printf("取出节点%d\n",temp.getId());
            if (temp.getNext() == first){ //说明已经遍历到最后一个了
                break;
            }
            temp = temp.getNext();
        }
    }

    //根据参数让节点出圈(Josepfu)
    public void josepfu(int startNode,int count,int num){ //startNode为开始的那个节点,count为每次数第几个,num为链表节点个数
        if (first == null || startNode < 1 || count < 1 || startNode > num){
            System.out.println("链表为空或者输入的参数不符合标准!");
            return;
        }
        //让first移动到startNode指定的节点,即移动startNode-1次
        for (int i = 0; i < startNode - 1; i++) {
            first = first.getNext();
        }
        //创建一个辅助变量,让其指向最后一个节点(first前一个)
        JNode helper = first;
        while (helper.getNext() != first){
            helper = helper.getNext();
        }
        //开始按照要求出圈,每次都让helper和first移动count-1次
        while (true){
            if (helper == first){ //圈中只剩下一个节点
                break;
            }
            for (int i = 0; i < count - 1; i++) {
                first = first.getNext();
                helper = helper.getNext();
            }
            //此时first指向的即为要出圈的节点
            System.out.printf("节点%d出圈\n",first.getId());
            //将出圈的节点从链表中移除
            first = first.getNext();
            helper.setNext(first);
        }
        System.out.printf("节点%d为最后一个节点",first.getId());
    }
}

测试类

/**
 * @Author: Yeman
 * @Date: 2021-10-15-22:33
 * @Description:
 */
public class JosepfuTest {
    public static void main(String[] args) {

        CircleSingleLinkedList linkedList = new CircleSingleLinkedList();

        linkedList.add(5);

        linkedList.list();
        System.out.println("===================");

        linkedList.josepfu(1,2,5);
    }
}

到此这篇关于Java用单向环形链表来解决约瑟夫环Josepfu问题的文章就介绍到这了,更多相关Java 单向环形链表 内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • java实现猜字母游戏

    java实现猜字母游戏

    这篇文章主要为大家详细介绍了java实现猜字母小游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-03-03
  • java中BIO、NIO、AIO都有啥区别

    java中BIO、NIO、AIO都有啥区别

    这篇文章主要介绍了java中BIO、NIO、AIO都有啥区别,IO模型就是说用什么样的通道进行数据的发送和接收,Java共支持3种网络编程IO模式:BIO,NIO,AIO,文中有非常详细的代码示例,对正在学习java的小伙伴们有非常好的帮助,需要的朋友可以参考下
    2021-04-04
  • SpringSecurity自定义成功失败处理器的示例代码

    SpringSecurity自定义成功失败处理器的示例代码

    这篇文章主要介绍了SpringSecurity自定义成功失败处理器,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-09-09
  • Java实现动态验证码生成

    Java实现动态验证码生成

    这篇文章主要为大家详细介绍了Java实现动态验证码生成,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-04-04
  • 浅谈IDEA2018打包可执行jar包的流程

    浅谈IDEA2018打包可执行jar包的流程

    这篇文章主要介绍了浅谈IDEA2018打包可执行jar包的流程,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-06-06
  • Java编程WeakHashMap实例解析

    Java编程WeakHashMap实例解析

    这篇文章主要介绍了Java编程WeakHashMap实例解析,分享了相关代码示例,小编觉得还是挺不错的,具有一定借鉴价值,需要的朋友可以参考下
    2018-02-02
  • spring boot配置druid连接池的完整步骤

    spring boot配置druid连接池的完整步骤

    这篇文章主要给大家介绍了关于spring boot配置druid连接池的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-01-01
  • 使用IDEA创建SpringBoot项目的方法步骤

    使用IDEA创建SpringBoot项目的方法步骤

    这篇文章主要介绍了使用IDEA创建SpringBoot项目的方法步骤,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-05-05
  • Java常用JVM参数实战

    Java常用JVM参数实战

    本文主要介绍了Java常用JVM参数实战,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-05-05
  • dubbo新手学习之事件通知实践教程

    dubbo新手学习之事件通知实践教程

    这篇文章主要给大家介绍了关于dubbo新手学习之事件通知实践的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-09-09

最新评论