php基于雙向循環(huán)隊(duì)列實(shí)現(xiàn)歷史記錄的前進(jìn)后退等功能
本文實(shí)例講述了php基于雙向循環(huán)隊(duì)列實(shí)現(xiàn)歷史記錄的前進(jìn)后退等功能。分享給大家供大家參考。具體如下:
為實(shí)現(xiàn)一個(gè)記錄操作歷史的功能
1. 和撤銷(xiāo),反撤銷(xiāo)功能類(lèi)似的一個(gè)功能。(實(shí)現(xiàn)操作的前進(jìn)后退)
2. 和discuz論壇登錄后查看帖子(可以前進(jìn)后退查看過(guò)的帖子,還有帖子查看歷史記錄)
3. 邏輯和windows資源管理器地址欄前進(jìn)后退功能一樣。
根據(jù)這種需要,實(shí)現(xiàn)了一個(gè)數(shù)據(jù)結(jié)構(gòu)。寫(xiě)了一個(gè)通用的類(lèi),暫叫歷史記錄類(lèi)吧。
【原理和時(shí)鐘類(lèi)似。實(shí)例化對(duì)象時(shí)可以構(gòu)造長(zhǎng)度為N(可以根據(jù)需要定長(zhǎng)度)個(gè)節(jié)點(diǎn)的環(huán)】
然后整合各種操作。前進(jìn)、后退、插入、修改插入。
類(lèi)可以構(gòu)造一個(gè)數(shù)組?;蛘邆魅霐?shù)組參數(shù)構(gòu)造一個(gè)對(duì)象。 每次操作之后可以取得操作后的數(shù)組。 操作完的 數(shù)據(jù)可以根據(jù)自己的需要以合適的方式保存。 放在cookie,session里面,或者序列化,或轉(zhuǎn)為json數(shù)據(jù)保存在數(shù)據(jù)庫(kù)里,或者放在文件里面都可以。 方便下一次使用。
為了便于擴(kuò)展,存放更多的數(shù)據(jù)。具體每一條數(shù)據(jù)也是一條數(shù)組記錄。
比如根據(jù)需要進(jìn)行擴(kuò)展:array('path'=>'D:/www/','sss'=>value)
順便貼出,自己寫(xiě)的調(diào)試變量用的一個(gè)文件。
1. pr()可以格式化并高亮輸出變量。pr($arr),pr($arr,1)是輸出后退出。
2. debug_out() 用來(lái)輸出多個(gè)變量。默認(rèn)為退出。
3. debug_out($_GET,$_SERVER,$_POST,$arr);
history.class.php文件:
<?php
include 'debug.php';
/**
* 歷史記錄操作類(lèi)
* 傳入或者構(gòu)造一個(gè)數(shù)組。形如:
array(
'history_num'=>20, //隊(duì)列節(jié)點(diǎn)總共個(gè)數(shù)
'first'=>0, //起始位置,從0開(kāi)始。數(shù)組索引值
'last'=>0, //終點(diǎn)位置,從0開(kāi)始。
'back'=>0, //從first位置倒退了多少步,差值。
'history'=>array( //數(shù)組,存放操作隊(duì)列。
array('path'=>'D:/'),
array('path'=>'D:/www/'),
array('path'=>'E:/'),
array('path'=>'/home/')
……
)
)
*/
class history{
var $history_num;
var $first;
var $last;
var $back;
var $history=array();
function __construct($array=array(),$num=12){
if (!$array) {//數(shù)組為空.構(gòu)造一個(gè)循環(huán)隊(duì)列。
$history=array();
for ($i=0; $i < $num; $i++) {
array_push($history,array('path'=>''));
}
$array=array(
'history_num'=>$num,
'first'=>0,//起始位置
'last'=>0,//終點(diǎn)位置
'back'=>0,
'history'=>$history
);
}
$this->history_num=$array['history_num'];
$this->first=$array['first'];
$this->last=$array['last'];
$this->back=$array['back'];
$this->history=$array['history'];
}
function nextNum($i,$n=1){//環(huán)路下n一個(gè)值。和時(shí)鐘環(huán)路類(lèi)似。
return ($i+$n)<$this->history_num ? ($i+$n):($i+$n-$this->history_num);
}
function prevNum($i,$n=1){//環(huán)路上一個(gè)值i?;赝薔個(gè)位置。
return ($i-$n)>=0 ? ($i-$n) : ($i-$n+$this->history_num);
}
function minus($i,$j){//順時(shí)針兩點(diǎn)只差,i-j
return ($i > $j) ? ($i - $j):($i-$j+$this->history_num);
}
function getHistory(){//返回?cái)?shù)組,用于保存或者序列化操作。
return array(
'history_num'=> $this->history_num,
'first' => $this->first,
'last' => $this->last,
'back' => $this->back,
'history' => $this->history
);
}
function add($path){
if ($this->back!=0) {//有后退操作記錄的情況下,進(jìn)行插入。
$this->goedit($path);
return;
}
if ($this->history[0]['path']=='') {//剛構(gòu)造,不用加一.首位不前移
$this->history[$this->first]['path']=$path;
return;
}else{
$this->first=$this->nextNum($this->first);//首位前移
$this->history[$this->first]['path']=$path;
}
if ($this->first==$this->last) {//起始位置與終止位置相遇
$this->last=$this->nextNum($this->last);//末尾位置前移。
}
}
function goback(){//返回從first后退N步的地址。
$this->back+=1;
//最大后退步數(shù)為起點(diǎn)到終點(diǎn)之差(順時(shí)針之差)
$mins=$this->minus($this->first,$this->last);
if ($this->back >= $mins) {//退到最后點(diǎn)
$this->back=$mins;
}
$pos=$this->prevNum($this->first,$this->back);
return $this->history[$pos]['path'];
}
function gonext(){//從first后退N步的地方前進(jìn)一步。
$this->back-=1;
if ($this->back<0) {//退到最后點(diǎn)
$this->back=0;
}
return $this->history[$this->prevNum($this->first,$this->back)]['path'];
}
function goedit($path){//后退到某個(gè)點(diǎn),沒(méi)有前進(jìn)而是修改。則firs值為最后的值。
$pos=$this->minus($this->first,$this->back);
$pos=$this->nextNum($pos);//下一個(gè)
$this->history[$pos]['path']=$path;
$this->first=$pos;
$this->back=0;
}
//是否可以后退
function isback(){
if ($this->back < $this->minus($this->first,$this->last)) {
return ture;
}
return false;
}
//是否可以前進(jìn)
function isnext(){
if ($this->back>0) {
return true;
}
return false;
}
}
//測(cè)試代碼。
$hi=new history(array(),6);//傳入空數(shù)組,則初始化數(shù)組構(gòu)造。
for ($i=0; $i <8; $i++) {
$hi->add('s'.$i);
}
pr($hi->goback());
pr($hi->goback());
pr($hi->goback());
pr($hi->gonext());
pr($hi->gonext());
pr($hi->gonext());
pr($hi->gonext());
$hi->add('asdfasdf');
$hi->add('asdfasdf2');
pr($hi->getHistory());
$ss=new history($hi->getHistory());//直接用數(shù)組構(gòu)造。
$ss->add('asdfasdf');
$ss->goback();
pr($ss->getHistory());
?>
debug.php文件:
<?php
/**
* 獲取變量的名字
* eg hello="123" 獲取ss字符串
*/
function get_var_name(&$aVar){
foreach($GLOBALS as $key=>$var)
{
if($aVar==$GLOBALS[$key] && $key!="argc"){
return $key;
}
}
}
/**
* 格式化輸出變量,或者對(duì)象
* @param mixed $var
* @param boolean $exit
*/
function pr($var,$exit = false){
ob_start();
$style='<style>
pre#debug{margin:10px;font-size:13px;color:#222;font-family:Consolas ;line-height:1.2em;background:#f6f6f6;border-left:5px solid #444;padding:5px;width:95%;word-break:break-all;}
pre#debug b{font-weight:400;}
#debug #debug_str{color:#E75B22;}
#debug #debug_keywords{font-weight:800;color:00f;}
#debug #debug_tag1{color:#22f;}
#debug #debug_tag2{color:#f33;font-weight:800;}
#debug #debug_var{color:#33f;}
#debug #debug_var_str{color:#f00;}
#debug #debug_set{color:#0C9CAE;}</style>';
if (is_array($var)){
print_r($var);
}
else if(is_object($var)){
echo get_class($var)." Object";
}
else if(is_resource($var)){
echo (string)$var;
}
else{
echo var_dump($var);
}
$out = ob_get_clean();//緩沖輸出給$out 變量
$out=preg_replace('/"(.*)"/','<b id="debug_var_str">"'.'\\1'.'"</b>',$out);//高亮字符串變量
$out=preg_replace('/=\>(.*)/','=>'.'<b id="debug_str">'.'\\1'.'</b>',$out);//高亮=>后面的值
$out=preg_replace('/\[(.*)\]/','<b id="debug_tag1">[</b><b id="debug_var">'.'\\1'.'</b><b id="debug_tag1">]</b>',$out);//高亮變量
$from = array(' ','(',')','=>');
$to = array(' ','<b id="debug_tag2">(</i>','<b id="debug_tag2">)</b>','<b id="debug_set">=></b>');
$out=str_replace($from,$to,$out);
$keywords=array('Array','int','string','class','object','null');//關(guān)鍵字高亮
$keywords_to=$keywords;
foreach($keywords as $key=>$val)
{
$keywords_to[$key] = '<b id="debug_keywords">'.$val.'</b>';
}
$out=str_replace($keywords,$keywords_to,$out);
echo $style.'<pre id="debug"><b id="debug_keywords">'.get_var_name($var).'</b> = '.$out.'</pre>';
if ($exit) exit;//為真則退出
}
/**
* 調(diào)試輸出變量,對(duì)象的值。
* 參數(shù)任意個(gè)(任意類(lèi)型的變量)
* @return echo
*/
function debug_out(){
$avg_num = func_num_args();
$avg_list= func_get_args();
ob_start();
for($i=0; $i < $avg_num; $i++) {
pr($avg_list[$i]);
}
$out=ob_get_clean();
echo $out;
exit;
}
?>
希望本文所述對(duì)大家的php程序設(shè)計(jì)有所幫助。
- PHP基于數(shù)組實(shí)現(xiàn)的堆棧和隊(duì)列功能示例
- 關(guān)于PHP堆棧與列隊(duì)的學(xué)習(xí)
- php線(xiàn)性表的入棧與出棧實(shí)例分析
- PHP基于堆棧實(shí)現(xiàn)的高級(jí)計(jì)算器功能示例
- PHP實(shí)現(xiàn)的棧數(shù)據(jù)結(jié)構(gòu)示例【入棧、出棧、遍歷棧】
- PHP實(shí)現(xiàn)基于棧的后綴表達(dá)式求值功能
- PHP使用數(shù)組實(shí)現(xiàn)隊(duì)列
- php實(shí)現(xiàn)的雙向隊(duì)列類(lèi)實(shí)例
- 隊(duì)列在編程中的實(shí)際應(yīng)用(php)
- PHP實(shí)現(xiàn)的鏈?zhǔn)疥?duì)列結(jié)構(gòu)示例
- PHP使用兩個(gè)棧實(shí)現(xiàn)隊(duì)列功能的方法
相關(guān)文章
Look And Say 序列php實(shí)現(xiàn)代碼
Look And Say序列,簡(jiǎn)單的說(shuō)就是根據(jù)你看到的數(shù)字,寫(xiě)出下一個(gè)數(shù)字2011-05-05
php圖像處理函數(shù)imagecopyresampled用法詳解
這篇文章主要介紹了php圖像處理函數(shù)imagecopyresampled用法,結(jié)合實(shí)例形式詳細(xì)分析了imagecopyresampled函數(shù)的功能、參數(shù)、使用方法,需要的朋友可以參考下2016-12-12
php實(shí)現(xiàn)微信公眾號(hào)主動(dòng)推送消息
這篇文章主要介紹了php實(shí)現(xiàn)微信公眾號(hào)主動(dòng)推送消息的方法,PHP版微信公共平臺(tái)消息主動(dòng)推送,突破訂閱號(hào)一天只能發(fā)送一條信息限制,需要的朋友可以參考下2015-12-12
在VSCode中配置PHP開(kāi)發(fā)環(huán)境的實(shí)戰(zhàn)步驟
最近要寫(xiě)一些可視化的網(wǎng)站,所以先把需要的環(huán)境配好吧,下面這篇文章主要給大家介紹了關(guān)于在VSCode中配置PHP開(kāi)發(fā)環(huán)境的相關(guān)資料,文中通過(guò)圖文介紹的非常詳細(xì),需要的朋友可以參考下2022-11-11
php數(shù)組函數(shù)序列之a(chǎn)rray_intersect() 返回兩個(gè)或多個(gè)數(shù)組的交集數(shù)組
array_intersect() 函數(shù)返回兩個(gè)或多個(gè)數(shù)組的交集數(shù)組。結(jié)果數(shù)組包含了所有在被比較數(shù)組中,也同時(shí)出現(xiàn)在所有其他參數(shù)數(shù)組中的值,鍵名保留不變。2011-11-11

