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

2019年2月25日17:29:17

Josephus有过的故事:39 个犹太人与Josephus及他的朋友躲到一个洞中,39个犹太人决定宁愿死也不要被敌人抓。于是决定了自杀方式,41个人排成一个圆圈,由第1个人开始报数,每报数到第3人该人就必须自杀。然后下一个重新报数,直到所有人都自杀身亡为止。然而Josephus 和他的朋友并不想遵从,Josephus要他的朋友先假装遵从,他将朋友与自己安排在第16个与第31个位置,于是逃过了这场死亡游戏。

抽象出的问题是 N个人围成一圈,从第S个人开始报数,报到C的人出圈,从出去的人的下一位,继续从下一任开始重新报数,报到m的人出圈;如此往复,直到所有人出圈。

final class Kid {

    public $no;
    public $next = null;

    public function __construct($no) {
        $this->no = $no;
    }

}
<?php

/*
 * 环形链表 解决约瑟夫问题
 */

final class CircularLinkedList {

    public function addKid($n = 0, &$head = null) {
        for ($i = 0; $i < $n; $i++) {
            $Kid = new Kid($i + 1);
            if ($i == 0) { //第一个小孩的情况
                $head = $Kid; //对象赋值,是引用赋值
                $head->next = $Kid; //自己指向自己
                $current = $head; //对象赋值,是引用赋值
            } else {
                $current->next = $Kid;
                $Kid->next = $head;
//                //继续指向下一个
                $current = $current->next;
            }
        }
    }

    /*
     * $start 从几开始
     * $count 数到几就出圈
     */

    public function play(Kid $head, $start, $count) {
        $current = $head;
        //移动指针从$start 移动到
        while (1) {
            if ($current->no == $start) {
                break;
            }
            $current = $current->next;
        }
//        p($current);
//        pp($this->countKids($current));
//        $all = $this->countKids($current);

        while ($current->next != $current->next->next) {
            //少移动一位,方便一处节点
            for ($i = 1; $i < $count; $i++) {
                $current = $current->next;
            }
            //去除节点
//            p($current);
            p('出去的小孩是  --' . $current->next->no);
            $current->next = $current->next->next;
//            p($current);
            //移动指针,移到删除节点的下一位就是重新数数的那个节点
            $current = $current->next;
        }
        p($current->no);
    }

    public function countKids(Kid $head) {
        $current = $head;
        $count = 1;
        while ($head->no != $current->next->no) {
            $count++;
            $current = $current->next;
        }

        return $count;
    }

}

调用

$CircularLinkedList = new CircularLinkedList();
$CircularLinkedList->addKid(10, $head);
$CircularLinkedList->play($head, 3, 2);

结果

出去的小孩是  --5
出去的小孩是  --8
出去的小孩是  --1
出去的小孩是  --4
出去的小孩是  --9
出去的小孩是  --3
出去的小孩是  --10
出去的小孩是  --7
出去的小孩是  --2
6

猜你喜欢

转载自www.cnblogs.com/zx-admin/p/10432136.html