php队列排序

新学期开始了,小哈是小哼的新同桌(小哈是个小美女哦~),小哼向小哈询问QQ号,小哈当然不会直接告诉小哼啦,原因嘛你懂的。所以小哈给了小哼一串加密过的数字,同时小哈也告诉了小哼解密规则。

规则是这样的:首先将第1个数删除,紧接着将第2个数放到这串数的末尾,再将第3个数删除并将第4个数再放到这串数的末尾,再将第5个数删除……直到剩下最后一个数,将最后一个数也删除。按照刚才删除的顺序,把这些删除的数连在一起就是小哈的QQ啦。现在你来帮帮小哼吧。小哈给小哼加密过的一串数是“6 3 1 7 5 8 9 2 4”。

OK,现在轮到你动手的时候了。快去找出9张便签或小纸片,将“6 3 1 7 5 8 9 2 4”这9个数分别写在9张便签上,模拟一下解密过程。如果你没有理解错解密规则的话,解密后小哈的QQ号应该是“6 1 5 9 4 7 2 8 3”。

其实解密的过程就像是将这些数“排队”。每次从最前面拿两个,第1个扔掉,第2个放到尾部。具体过程是这样的:刚开始这串数是“6 3 1 7 5 8 9 2 4”,首先删除6并将3放到这串数的末尾,这串数更新为“1 7 5 8 9 2 4 3”。接下来删除1并将7放到末尾,即更新为“5 8 9 2 4 3 7”。再删除5并将8放到末尾即“9 2 4 3 7 8”,删除9并将2放到末尾即“4 3 7 8 2”,删除4并将3放到末尾即“7 8 2 3”,删除7并将8放到末尾即“2 3 8”,删除2并将3放到末尾即“8 3”,删除8并将3放到末尾即“3”,最后删除3。因此被删除的顺序是“6 1 5 9 4 7 2 8 3”,这就是小哈的QQ号码了,你可以加她试试看^_^。也许真的有……………..

如果用把它编成程序最简单的肯定是这样:

function queue($string){
    $result = "";
    $len = strlen($string);
    for($i=0;$i<$len;$i++){
        $result .= substr($string,0,1);                    //取出第一个数
        $string = substr($string,2).substr($string,1,1);   //重组新的字符串
    }
    return $result;
}
echo(queue("631758924"));

这就是按照上面的思想:取出第一个数,然后构建新的字符串再循环。

我们可以把它模拟成一个排队买票的队伍,最前面的人买好离开了,后面所有的人就需要全部向前面走一步,补上之前的空位。第二个人买好离开,同时队尾又新加了一个人,在上面的情景中第二个人跟队尾的新人是同一个(请忽略为毛第二个买好还要到队尾)。同时形成一个新的队伍,然后按照刚刚的方式循环。
这样就可以引入两个变量head和tail记录队首(即第一位)跟队尾(即最后一位),这里规定队首和队尾重合时,队列为空。
现在有9个数,9个数全部放入队列之后head=0;tail=8;此时head和tail之间的数就是目前队列中“有效”的数。如果要删除一个数的话,就将head++就OK了,这样仍然可以保持head和tail之间的数为目前队列中“有效”的数。这样做虽然浪费了一个空间,却节省了大量的时间,这是非常划算的。新增加一个数也很简单,把需要增加的数放到队尾之后再tail++就欧克啦。不多说上代码:

function queue($string){
    $arr = str_split($string,1);         //构建队列
    $head = 0;
    $tail = count($arr)-1;
    $result = $arr[$head];                   //先取出第一个
    while($head<$tail){
        $head++;
        $arr[$tail+1] = $arr[$head];          //队尾添加一个数为第二个
        $tail++;
        $head++;
        $result .= $arr[$head];               //再取出第二个队列的第一个      
    }
    return $result;
}

我们再来总结一下队列的概念。队列是一种特殊的线性结构,它只允许在队列的首部(head)进行删除操作称之为“出队”,而在队列的尾部(tail)进行插入操作称之为“入队”。当队列中没有元素时(即head==tail),称为空队列。在我们的日常生活中有很多情况都符合队列的特性。比如我们之前提到过的买票,每个排队买票的窗口就是一个队列。在这个队列当中,新来的人总是站在队列的最后面,来的越早的人越靠前也就越早能买到票,就是先来的人先服务,我们称为“先进先出”(First InFirst Out,FIFO)原则。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值